Compare commits
	
		
			627 Commits
		
	
	
		
			2014.03.23
			...
			2014.06.26
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 4242001863 | ||
|   | 78338f71ca | ||
|   | f5172a3084 | ||
|   | c7df67edbd | ||
|   | d410fee91d | ||
|   | ba7aa464de | ||
|   | 8333034dce | ||
|   | 637b6af80f | ||
|   | 1044f8afd2 | ||
|   | 2f775107f9 | ||
|   | 85342674b2 | ||
|   | fd69098a45 | ||
|   | 8867f908fc | ||
|   | b7c33124c8 | ||
|   | 89a8c423c7 | ||
|   | cea2582df2 | ||
|   | e423e0baaa | ||
|   | 60b2dd1285 | ||
|   | 36ddd8b3f7 | ||
|   | 7575d52a73 | ||
|   | 9a2dc4f7ac | ||
|   | c5cd249e41 | ||
|   | 8940c1c058 | ||
|   | 27ec04b232 | ||
|   | d2824416aa | ||
|   | 18061bbab0 | ||
|   | 4ecbbcbcea | ||
|   | 55c97a03e1 | ||
|   | 98aeac6ea9 | ||
|   | 8bfb6723cb | ||
|   | a20575e8ae | ||
|   | 7724572519 | ||
|   | d763637f6a | ||
|   | c26e9ac4b2 | ||
|   | 896bf55352 | ||
|   | a23ba9b53c | ||
|   | 38a9339baf | ||
|   | def8b4039f | ||
|   | a14e1538fe | ||
|   | 5f28a1acad | ||
|   | 25e9953c6f | ||
|   | f9df094ca5 | ||
|   | b60a469023 | ||
|   | 7012631257 | ||
|   | e6c9f80c48 | ||
|   | 895ce482b1 | ||
|   | e5da4021eb | ||
|   | 2371053565 | ||
|   | 33bf9033e0 | ||
|   | 35eacd0dae | ||
|   | 96bef88f5f | ||
|   | 5524b242a7 | ||
|   | a013eba65f | ||
|   | 36755d40b4 | ||
|   | 7d568f5ab8 | ||
|   | a7207cd580 | ||
|   | e8ef659cd9 | ||
|   | b0adbe98fb | ||
|   | 0c361c41b8 | ||
|   | c5469e046a | ||
|   | 4d2f143ce5 | ||
|   | 8f93030c85 | ||
|   | fdb9aebead | ||
|   | 3141feb73b | ||
|   | 9706f3f802 | ||
|   | d5e944359e | ||
|   | 826ec77fb2 | ||
|   | 2656f4eb6a | ||
|   | 2b88feedf7 | ||
|   | 23566e0d78 | ||
|   | 828553b614 | ||
|   | 3048e82a94 | ||
|   | 09ffa08ba1 | ||
|   | e0b4cc489f | ||
|   | 15e423407f | ||
|   | 702e522044 | ||
|   | 77abae55df | ||
|   | 617c0b2239 | ||
|   | 814d4257df | ||
|   | 23ae281b31 | ||
|   | 94128d6b0d | ||
|   | 059009c592 | ||
|   | 9cc977f104 | ||
|   | 1c0ade7afa | ||
|   | f2741c8d3a | ||
|   | 6ab8f3584a | ||
|   | 8ae5ce1726 | ||
|   | eb92077720 | ||
|   | 90e0fd4bad | ||
|   | 05741e05d9 | ||
|   | 9aa6637644 | ||
|   | d30d28156d | ||
|   | be6d722904 | ||
|   | d551980823 | ||
|   | f0a6c3d2bc | ||
|   | 4e0fb1280a | ||
|   | 24f5251cce | ||
|   | ac1390eee8 | ||
|   | 4a5b4d34dc | ||
|   | 63adb0cc61 | ||
|   | 3c80377b69 | ||
|   | 24577db241 | ||
|   | 566bd96da8 | ||
|   | ebdb64d605 | ||
|   | a6ffb92f0b | ||
|   | 3217377b3c | ||
|   | 24da5893fc | ||
|   | 087ca2cb07 | ||
|   | b4e7447458 | ||
|   | a45e6aadd7 | ||
|   | 70e322695d | ||
|   | 6a15923b77 | ||
|   | 7ffad0af5a | ||
|   | 0e3ae92441 | ||
|   | b3ae826f7a | ||
|   | dede691aca | ||
|   | fb6a5b965b | ||
|   | 6340716b3a | ||
|   | b675b32e6b | ||
|   | 6a3fa81ffb | ||
|   | df53a98f2b | ||
|   | db23d8d2a2 | ||
|   | 0d69795014 | ||
|   | 3374f3fdc2 | ||
|   | 4bf0727b1f | ||
|   | 263bd4ec50 | ||
|   | b7e8b6e37a | ||
|   | ceb7a17f34 | ||
|   | 1a2f2e1e66 | ||
|   | 6803016858 | ||
|   | 9b7c4fd981 | ||
|   | dc31942f42 | ||
|   | 1f6b8f3115 | ||
|   | 9c7b79acd9 | ||
|   | 9168308579 | ||
|   | 7e8fdb1aae | ||
|   | 386ba39cac | ||
|   | 236d0cd07c | ||
|   | ed86f38a11 | ||
|   | 6db80ad2db | ||
|   | 14470ac87b | ||
|   | 0cdf576d86 | ||
|   | 4ffeca4ea2 | ||
|   | 211fd6c674 | ||
|   | 6ebb46c106 | ||
|   | 0f97c9a06f | ||
|   | 77fb72646f | ||
|   | aae74e3832 | ||
|   | 894e730911 | ||
|   | 63961d87a6 | ||
|   | 87fe568c28 | ||
|   | 46531b374d | ||
|   | 9e8753911c | ||
|   | 5c6b1e578c | ||
|   | 8f0c8fb452 | ||
|   | b702ecebf0 | ||
|   | 950dc95e97 | ||
|   | d9dd3584e1 | ||
|   | 15a9f36849 | ||
|   | d0087d4ff2 | ||
|   | cc5ada6f4c | ||
|   | dfb2e1a325 | ||
|   | 65bab327b4 | ||
|   | 9eeb7abc6b | ||
|   | c70df21099 | ||
|   | 418424e5f5 | ||
|   | 8477466125 | ||
|   | 865dbd4a26 | ||
|   | b1e6f55912 | ||
|   | 4d78f3b770 | ||
|   | 7f739999e9 | ||
|   | 0f8a01d4f3 | ||
|   | e2bf499b14 | ||
|   | 7cf4547ab6 | ||
|   | 8ae980807a | ||
|   | eec4d8ef96 | ||
|   | 1c783bca88 | ||
|   | ac73651f66 | ||
|   | e5ceb3bfda | ||
|   | c2ef29234c | ||
|   | 1a1826c1af | ||
|   | c7c6d43fe1 | ||
|   | 2902d44f99 | ||
|   | d6e4ba287b | ||
|   | f50ee8d1c3 | ||
|   | 0e67ab0d8e | ||
|   | 77541837e5 | ||
|   | e3a6576f35 | ||
|   | 89bb8e97ee | ||
|   | 375696b1b1 | ||
|   | 4ea5c7b70d | ||
|   | 8dfa187b8a | ||
|   | c1ed1f7055 | ||
|   | 1514f74967 | ||
|   | 2e8323e3f7 | ||
|   | 69f8364042 | ||
|   | 79981f039b | ||
|   | 34d863f3fc | ||
|   | 91994c2c81 | ||
|   | 3ee4b60d56 | ||
|   | 76e92371ac | ||
|   | 08af0205f9 | ||
|   | a725fb1f43 | ||
|   | 05ee2b6dad | ||
|   | b74feacac5 | ||
|   | 426b52fc5d | ||
|   | 5c30b26846 | ||
|   | f07b74fc18 | ||
|   | a5a45015ba | ||
|   | beee53de06 | ||
|   | 8712f2bea7 | ||
|   | ea102818c9 | ||
|   | 0a871f6880 | ||
|   | 481efc84a8 | ||
|   | 01ed5c9be3 | ||
|   | ad3bc6acd5 | ||
|   | 5afa7f8bee | ||
|   | ec8deefc27 | ||
|   | a2d5a4ee64 | ||
|   | dffcc2ea0c | ||
|   | 1800eeefed | ||
|   | d7e7dedbde | ||
|   | d19bb9c0aa | ||
|   | 3ef79a974a | ||
|   | bc6800fbed | ||
|   | 65314dccf8 | ||
|   | feb7221209 | ||
|   | 56a94d8cbb | ||
|   | 24e6ec8ac8 | ||
|   | 87724af7a8 | ||
|   | b65c3e77e8 | ||
|   | 5301304bf2 | ||
|   | 948bcc60df | ||
|   | 25dfe0eb10 | ||
|   | 8e71456a81 | ||
|   | ccdd34ed78 | ||
|   | 26d886354f | ||
|   | a172b258ac | ||
|   | 7b93c2c204 | ||
|   | 57c7411f46 | ||
|   | d0a122348e | ||
|   | e4cbb5f382 | ||
|   | c1bce22f23 | ||
|   | e3abbbe301 | ||
|   | 55b36e3710 | ||
|   | 877bea9ce1 | ||
|   | 33c7ff861e | ||
|   | 749fe60c1e | ||
|   | 63b31b059c | ||
|   | 1476b497eb | ||
|   | e399853d0c | ||
|   | fdb205b19e | ||
|   | fbe8053120 | ||
|   | ea783d01e1 | ||
|   | b7d73595dc | ||
|   | e97e53eeed | ||
|   | 342f630dbf | ||
|   | 69c8fb9e5d | ||
|   | 5f0f8013ac | ||
|   | b5368acee8 | ||
|   | f71959fcf5 | ||
|   | 5c9f3b8b16 | ||
|   | bebd6f9308 | ||
|   | 84a2806c16 | ||
|   | d0111a7409 | ||
|   | aab8874c55 | ||
|   | fcf5b01746 | ||
|   | 4de9e9a6db | ||
|   | 0067d6c4be | ||
|   | 2099125333 | ||
|   | b48f147d5a | ||
|   | 4f3e943080 | ||
|   | 7558830fa3 | ||
|   | 867274e997 | ||
|   | 6515778305 | ||
|   | 3b1dfc0f2f | ||
|   | d664de44b7 | ||
|   | bbe99d26ec | ||
|   | 50fc59968e | ||
|   | b8b01bb92a | ||
|   | eb45133451 | ||
|   | 10c0e2d818 | ||
|   | 669f0e7cda | ||
|   | 32fd27ec98 | ||
|   | 0c13f378de | ||
|   | 0049594efb | ||
|   | 113c7d3eb0 | ||
|   | 549371fc99 | ||
|   | 957f27e5bb | ||
|   | 1f8c19767b | ||
|   | a383a98af6 | ||
|   | acd69589a5 | ||
|   | b30b8698ea | ||
|   | f1f25be6db | ||
|   | deab8c1960 | ||
|   | c57f775710 | ||
|   | e75cafe9fb | ||
|   | 33ab8453c4 | ||
|   | ebd3c7b370 | ||
|   | 29645a1d44 | ||
|   | 22d99a801a | ||
|   | 57b8d84cd9 | ||
|   | 65e4ad5bfe | ||
|   | 98b7d476d9 | ||
|   | 201e3c99b9 | ||
|   | 8a7a4a9796 | ||
|   | df297c8794 | ||
|   | 3f53a75f02 | ||
|   | 7c360e3a04 | ||
|   | d2176c8011 | ||
|   | aa92f06308 | ||
|   | e00c9cf599 | ||
|   | ba60a3ebe0 | ||
|   | efb7e11988 | ||
|   | a55c8b7aac | ||
|   | a980bc4324 | ||
|   | 4b10aadffc | ||
|   | 5bec574859 | ||
|   | d11271dd29 | ||
|   | 1d9d26d09b | ||
|   | c0292e8ab7 | ||
|   | f44e5d8b43 | ||
|   | 6ea74538e3 | ||
|   | 24b8924b46 | ||
|   | 86a3c67112 | ||
|   | 8be874370d | ||
|   | aec74dd95a | ||
|   | 6890574256 | ||
|   | d03745c684 | ||
|   | 28746fbd59 | ||
|   | 0321213c11 | ||
|   | 3f0aae4244 | ||
|   | 48099643cc | ||
|   | 621f33c9d0 | ||
|   | f07a9f6f43 | ||
|   | e51880fd32 | ||
|   | 88ce273da4 | ||
|   | b9ba5dfa28 | ||
|   | 4086f11929 | ||
|   | 478c2c6193 | ||
|   | d2d6481afb | ||
|   | 43acb120f3 | ||
|   | e8f2025edf | ||
|   | a4eb9578af | ||
|   | fa35cdad02 | ||
|   | d1b9c912a4 | ||
|   | edec83a025 | ||
|   | c0a7c60815 | ||
|   | 117a7d1944 | ||
|   | a40e0dd434 | ||
|   | 188b086dd9 | ||
|   | 1f27d2c0e1 | ||
|   | 7560096db5 | ||
|   | 282cb9c7ba | ||
|   | 3a9d6790ad | ||
|   | 0610a3e0b2 | ||
|   | 7f9c31df88 | ||
|   | 3fa6b6e293 | ||
|   | 3c50b99ab4 | ||
|   | 52fadd5fb2 | ||
|   | 5367fe7f4d | ||
|   | 427588f6e7 | ||
|   | 51745be312 | ||
|   | d7f1e7c88f | ||
|   | 4145a257be | ||
|   | 525dc9809e | ||
|   | 1bf3210816 | ||
|   | e6c6d10d99 | ||
|   | f270256e06 | ||
|   | f401c6f69f | ||
|   | b075d25bed | ||
|   | 3d1bb6b4dd | ||
|   | 1db2666916 | ||
|   | 8f5c0218d8 | ||
|   | d7666dff82 | ||
|   | 2d4c98dbd1 | ||
|   | fd50bf623c | ||
|   | d360a14678 | ||
|   | d0f2ab6969 | ||
|   | de906ef543 | ||
|   | 2fb3deeca1 | ||
|   | 66398056f1 | ||
|   | 77477fa4c9 | ||
|   | a169e18ce1 | ||
|   | 381640e3ac | ||
|   | 37e3410137 | ||
|   | 97b5196960 | ||
|   | 6a4f3528c8 | ||
|   | b9c76aa1a9 | ||
|   | 0d3070d364 | ||
|   | 7753cadbfa | ||
|   | 3950450342 | ||
|   | c82b1fdad6 | ||
|   | b0fb63abe8 | ||
|   | 3ab34c603e | ||
|   | 7d6413341a | ||
|   | 140012d0f6 | ||
|   | 4be9f8c814 | ||
|   | 5c802bac37 | ||
|   | 6c30ff756a | ||
|   | 62749e4708 | ||
|   | 6b7dee4b38 | ||
|   | ef2041eb4e | ||
|   | 29e3e682af | ||
|   | f983c44199 | ||
|   | e4db19511a | ||
|   | c47d21da80 | ||
|   | 269aecd0c0 | ||
|   | aafddb2b0a | ||
|   | 6262ac8ac5 | ||
|   | 89938c719e | ||
|   | ec0fafbb19 | ||
|   | a5863bdf33 | ||
|   | b58ddb32ba | ||
|   | b9e12a8140 | ||
|   | 104aa7388a | ||
|   | c3855d28b0 | ||
|   | 734f90bb41 | ||
|   | 91a6addeeb | ||
|   | 9afb76c5ad | ||
|   | dfb2cb5cfd | ||
|   | 650d688d10 | ||
|   | 0ba77818f3 | ||
|   | 09baa7da7e | ||
|   | 85e787f51d | ||
|   | 2a9e1e453a | ||
|   | ee1e199685 | ||
|   | 17c5a00774 | ||
|   | 15c0e8e7b2 | ||
|   | cca37fba48 | ||
|   | 9d0993ec4a | ||
|   | 342f33bf9e | ||
|   | 7cd3bc5f99 | ||
|   | 931055e6cb | ||
|   | d0e4cf82f1 | ||
|   | 6f88df2c57 | ||
|   | 4479bf2762 | ||
|   | 1ff7c0f7d8 | ||
|   | 610e47c87e | ||
|   | 50f566076f | ||
|   | 92810ff497 | ||
|   | 60ccc59a1c | ||
|   | 91745595d3 | ||
|   | d6e40507d0 | ||
|   | deed48b472 | ||
|   | e4d41bfca5 | ||
|   | a355b70f27 | ||
|   | f8514f6186 | ||
|   | e09b8fcd9d | ||
|   | 7d1b527ff9 | ||
|   | f943c7b622 | ||
|   | 676eb3f2dd | ||
|   | 98b7cf1ace | ||
|   | c465afd736 | ||
|   | b84d6e7fc4 | ||
|   | 2efd5d78c1 | ||
|   | c8edf47b3a | ||
|   | 3b4c26a428 | ||
|   | 1525148114 | ||
|   | 9e0c5791c1 | ||
|   | 29a1ab2afc | ||
|   | fa387d2d99 | ||
|   | 6d0d573eca | ||
|   | bb799e811b | ||
|   | 04ee53eca1 | ||
|   | 659eb98a53 | ||
|   | ca6aada48e | ||
|   | 43df5a7e71 | ||
|   | 88f1c6de7b | ||
|   | 65a40ab82b | ||
|   | 4b9cced103 | ||
|   | 5c38625259 | ||
|   | 6344fa04bb | ||
|   | e3ced9ed61 | ||
|   | 5075d598bc | ||
|   | 68eb8e90e6 | ||
|   | d3a96346c4 | ||
|   | 0e518e2fea | ||
|   | 1e0a235f39 | ||
|   | 9ad400f75e | ||
|   | 3537b93d8a | ||
|   | 56eca2e956 | ||
|   | 2ad4d1ba07 | ||
|   | 4853de808b | ||
|   | 6ff5f12218 | ||
|   | 52a180684f | ||
|   | b21e25702f | ||
|   | 983af2600f | ||
|   | f34e6a2cd6 | ||
|   | a9f304031b | ||
|   | 9271bc8355 | ||
|   | d1b3e3dd75 | ||
|   | 968ed2a777 | ||
|   | 24de5d2556 | ||
|   | d26e981df4 | ||
|   | e45d40b171 | ||
|   | 4a419b8851 | ||
|   | 5fbd672c38 | ||
|   | bec1fad223 | ||
|   | 177fed41bc | ||
|   | b900e7cba4 | ||
|   | 14cb4979f0 | ||
|   | 69e61e30fe | ||
|   | cce929eaac | ||
|   | b6cfde99b7 | ||
|   | 1be99f052d | ||
|   | 2410c43d83 | ||
|   | aea6e7fc3c | ||
|   | 91a76c40c0 | ||
|   | d2b194607c | ||
|   | f6177462db | ||
|   | 9ddaf4ef8c | ||
|   | 97b5573848 | ||
|   | 18c95c1ab0 | ||
|   | 0479c625a4 | ||
|   | f659951e22 | ||
|   | 5853a7316e | ||
|   | a612753db9 | ||
|   | c8fc3fb524 | ||
|   | 5912c639df | ||
|   | 017e4dd58c | ||
|   | 651486621d | ||
|   | 28d9032c88 | ||
|   | 16f4eb723a | ||
|   | 1cbd410620 | ||
|   | d41ac5f5dc | ||
|   | 9c1fc022ae | ||
|   | 83d548ef0f | ||
|   | c72477bd32 | ||
|   | 9a7b072e38 | ||
|   | cbc4a6cc7e | ||
|   | cd7481a39e | ||
|   | acd213ed6d | ||
|   | 77ffa95701 | ||
|   | 2b25cb5d76 | ||
|   | 62fec3b2ff | ||
|   | e79162558e | ||
|   | 2da67107ee | ||
|   | 2ff7f8975e | ||
|   | 87a2566048 | ||
|   | 986f56736b | ||
|   | 2583a0308b | ||
|   | 40c716d2a2 | ||
|   | 79bfd01001 | ||
|   | f2bcdd8e02 | ||
|   | 8c5850eeb4 | ||
|   | bd3e077a2d | ||
|   | 7e70ac36b3 | ||
|   | 2cc0082dc0 | ||
|   | 056b56688a | ||
|   | b17418313f | ||
|   | e9a6fd6a68 | ||
|   | bf30f3bd9d | ||
|   | 330edf2d84 | ||
|   | 43f775e4ca | ||
|   | 8f6562448c | ||
|   | 263f4b514b | ||
|   | f0da3f1ef9 | ||
|   | cb3ac1c610 | ||
|   | 8efd15f477 | ||
|   | d26ebe990f | ||
|   | 28acf5500a | ||
|   | 214c22c704 | ||
|   | 8cdafb47b9 | ||
|   | 0dae5083f1 | ||
|   | 4c89bbd22c | ||
|   | e2b06e76c1 | ||
|   | e9c076c317 | ||
|   | 6c072e7d25 | ||
|   | ac6c104871 | ||
|   | 69c01a9f68 | ||
|   | e55213ce35 | ||
|   | 24a2aac445 | ||
|   | 784763c565 | ||
|   | 39c68260c0 | ||
|   | 149254d0d5 | ||
|   | 0c14e2fbe3 | ||
|   | 98acdc895b | ||
|   | bd3b5b8b10 | ||
|   | 9a90636805 | ||
|   | 6a66ae96ed | ||
|   | 2c8a4ba6b5 | ||
|   | ad8915b729 | ||
|   | 34cbc7ee8d | ||
|   | a59e40a1ea | ||
|   | ad0a75db6b | ||
|   | 1d0e49e1c7 | ||
|   | b4461b6ebe | ||
|   | 80959224fe | ||
|   | 865cbf4fc5 | ||
|   | 196f061cac | ||
|   | 99b380c33b | ||
|   | 02e4482e22 | ||
|   | b8a792de80 | ||
|   | fac55558ad | ||
|   | b2799ff96d | ||
|   | 7a249480b4 | ||
|   | f605128d13 | ||
|   | ba40a74666 | ||
|   | fb8ae2d438 | ||
|   | 893f8832b5 | ||
|   | 878d11ec29 | ||
|   | 515bbe4b5b | ||
|   | 75f2e25ba9 | ||
|   | 0d466d34a3 | ||
|   | 6949d81095 | ||
|   | f847ca02d3 | ||
|   | 510243ba58 | ||
|   | b540697a8a | ||
|   | 0d3641e589 | ||
|   | 72546c831e | ||
|   | d26db9269d | ||
|   | 4c0941853a | ||
|   | c11726364e | ||
|   | c577d735c6 | ||
|   | 9f0375f61a | ||
|   | 5e114e4bfe | ||
|   | 83622b6d2f | ||
|   | 3d87426c2d | ||
|   | ce328530a9 | ||
|   | f70daac108 | ||
|   | 912b38b428 | ||
|   | 6e25c58ed7 | ||
|   | 51fb2e98d2 | ||
|   | 38d63d846e | ||
|   | 410afb2003 | ||
|   | 685052fc7b | 
| @@ -3,6 +3,7 @@ python: | ||||
|   - "2.6" | ||||
|   - "2.7" | ||||
|   - "3.3" | ||||
|   - "3.4" | ||||
| script: nosetests test --verbose | ||||
| notifications: | ||||
|   email: | ||||
|   | ||||
							
								
								
									
										14
									
								
								CHANGELOG
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								CHANGELOG
									
									
									
									
									
								
							| @@ -1,14 +0,0 @@ | ||||
| 2013.01.02  Codename: GIULIA | ||||
|  | ||||
|     * Add support for ComedyCentral clips <nto> | ||||
|     * Corrected Vimeo description fetching <Nick Daniels> | ||||
|     * Added the --no-post-overwrites argument <Barbu Paul - Gheorghe> | ||||
|     * --verbose offers more environment info | ||||
|     * New info_dict field: uploader_id | ||||
|     * New updates system, with signature checking | ||||
|     * New IEs: NBA, JustinTV, FunnyOrDie, TweetReel, Steam, Ustream | ||||
|     * Fixed IEs: BlipTv | ||||
|     * Fixed for Python 3 IEs: Xvideo, Youku, XNXX, Dailymotion, Vimeo, InfoQ | ||||
|     * Simplified IEs and test code | ||||
|     * Various (Python 3 and other) fixes | ||||
|     * Revamped and expanded tests | ||||
| @@ -3,3 +3,4 @@ include test/*.py | ||||
| include test/*.json | ||||
| include youtube-dl.bash-completion | ||||
| include youtube-dl.1 | ||||
| recursive-include docs Makefile conf.py *.rst | ||||
|   | ||||
							
								
								
									
										11
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,7 +1,7 @@ | ||||
| all: youtube-dl README.md README.txt youtube-dl.1 youtube-dl.bash-completion | ||||
|  | ||||
| clean: | ||||
| 	rm -rf youtube-dl.1 youtube-dl.bash-completion README.txt MANIFEST build/ dist/ .coverage cover/ youtube-dl.tar.gz | ||||
| 	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 | ||||
|  | ||||
| cleanall: clean | ||||
| 	rm -f youtube-dl youtube-dl.exe | ||||
| @@ -55,7 +55,9 @@ README.txt: README.md | ||||
| 	pandoc -f markdown -t plain README.md -o README.txt | ||||
|  | ||||
| youtube-dl.1: README.md | ||||
| 	pandoc -s -f markdown -t man README.md -o youtube-dl.1 | ||||
| 	python devscripts/prepare_manpage.py >youtube-dl.1.temp.md | ||||
| 	pandoc -s -f markdown -t man youtube-dl.1.temp.md -o youtube-dl.1 | ||||
| 	rm -f youtube-dl.1.temp.md | ||||
|  | ||||
| youtube-dl.bash-completion: youtube_dl/*.py youtube_dl/*/*.py devscripts/bash-completion.in | ||||
| 	python devscripts/bash-completion.py | ||||
| @@ -72,8 +74,9 @@ youtube-dl.tar.gz: youtube-dl README.md README.txt youtube-dl.1 youtube-dl.bash- | ||||
| 		--exclude '__pycache' \ | ||||
| 		--exclude '.git' \ | ||||
| 		--exclude 'testdata' \ | ||||
| 		--exclude 'docs/_build' \ | ||||
| 		-- \ | ||||
| 		bin devscripts test youtube_dl \ | ||||
| 		CHANGELOG LICENSE README.md README.txt \ | ||||
| 		bin devscripts test youtube_dl docs \ | ||||
| 		LICENSE README.md README.txt \ | ||||
| 		Makefile MANIFEST.in youtube-dl.1 youtube-dl.bash-completion setup.py \ | ||||
| 		youtube-dl | ||||
|   | ||||
							
								
								
									
										93
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										93
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,11 +1,24 @@ | ||||
| % YOUTUBE-DL(1) | ||||
|  | ||||
| # NAME | ||||
| youtube-dl - download videos from youtube.com or other video platforms | ||||
|  | ||||
| # SYNOPSIS | ||||
| **youtube-dl** [OPTIONS] URL [URL...] | ||||
|  | ||||
| # INSTALLATION | ||||
|  | ||||
| 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 | ||||
|  | ||||
| If you do not have curl, you can alternatively use a recent wget: | ||||
|  | ||||
|     sudo wget https://yt-dl.org/downloads/2014.05.13/youtube-dl -O /usr/local/bin/youtube-dl | ||||
|     sudo chmod a+x /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). | ||||
|  | ||||
| Alternatively, refer to the developer instructions below for how to check out and work with the git repository. For further options, including PGP signatures, see https://rg3.github.io/youtube-dl/download.html . | ||||
|  | ||||
| # DESCRIPTION | ||||
| **youtube-dl** is a small command-line program to download videos from | ||||
| YouTube.com and a few more sites. It requires the Python interpreter, version | ||||
| @@ -28,6 +41,9 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|     --user-agent UA                  specify a custom user agent | ||||
|     --referer REF                    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 | ||||
|     --list-extractors                List all supported extractors and the URLs | ||||
|                                      they would handle | ||||
|     --extractor-descriptions         Output descriptions of all supported | ||||
| @@ -62,6 +78,7 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|                                      configuration in ~/.config/youtube-dl.conf | ||||
|                                      (%APPDATA%/youtube-dl/config.txt on | ||||
|                                      Windows) | ||||
|     --encoding ENCODING              Force the specified encoding (experimental) | ||||
|  | ||||
| ## Video Selection: | ||||
|     --playlist-start NUMBER          playlist video to start at (default is 1) | ||||
| @@ -166,6 +183,7 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|  | ||||
| ## Verbosity / Simulation Options: | ||||
|     -q, --quiet                      activates 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 | ||||
| @@ -177,7 +195,9 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|     --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 | ||||
|     -j, --dump-json                  simulate, quiet but print JSON information. | ||||
|                                      See --output for a description of available | ||||
|                                      keys. | ||||
|     --newline                        output progress bar as new lines | ||||
|     --no-progress                    do not print progress bar | ||||
|     --console-title                  display progress in console titlebar | ||||
| @@ -243,6 +263,7 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|                                      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) | ||||
| @@ -364,7 +385,67 @@ If you want to create a build of youtube-dl yourself, you'll need | ||||
|  | ||||
| ### Adding support for a new site | ||||
|  | ||||
| If you want to add support for a new site, copy *any* [recently modified](https://github.com/rg3/youtube-dl/commits/master/youtube_dl/extractor) file in `youtube_dl/extractor`, add an import in [`youtube_dl/extractor/__init__.py`](https://github.com/rg3/youtube-dl/blob/master/youtube_dl/extractor/__init__.py). Have a look at [`youtube_dl/common/extractor/common.py`](https://github.com/rg3/youtube-dl/blob/master/youtube_dl/extractor/common.py) for possible helper methods and a [detailed description of what your extractor should return](https://github.com/rg3/youtube-dl/blob/master/youtube_dl/extractor/common.py#L38). Don't forget to run the tests with `python test/test_download.py TestDownload.test_YourExtractor`! For a detailed tutorial, refer to [this blog post](http://filippo.io/add-support-for-a-new-video-site-to-youtube-dl/). | ||||
| If you want to add support for a new site, you can follow this quick list (assuming your service is called `yourextractor`): | ||||
|  | ||||
| 1. [Fork this repository](https://github.com/rg3/youtube-dl/fork) | ||||
| 2. Check out the source code with `git clone git@github.com:YOUR_GITHUB_USERNAME/youtube-dl.git` | ||||
| 3. Start a new git branch with `cd youtube-dl; git checkout -b yourextractor` | ||||
| 4. Start with this simple template and save it to `youtube_dl/extractor/yourextractor.py`: | ||||
|  | ||||
|         # coding: utf-8 | ||||
|         from __future__ import unicode_literals | ||||
|  | ||||
|         import re | ||||
|  | ||||
|         from .common import InfoExtractor | ||||
|          | ||||
|          | ||||
|         class YourExtractorIE(InfoExtractor): | ||||
|             _VALID_URL = r'https?://(?:www\.)?yourextractor\.com/watch/(?P<id>[0-9]+)' | ||||
|             _TEST = { | ||||
|                 'url': 'http://yourextractor.com/watch/42', | ||||
|                 'md5': 'TODO: md5 sum of the first 10KiB of the video file', | ||||
|                 'info_dict': { | ||||
|                     'id': '42', | ||||
|                     'ext': 'mp4', | ||||
|                     'title': 'Video title goes here', | ||||
|                     # TODO more properties, either as: | ||||
|                     # * A value | ||||
|                     # * MD5 checksum; start the string with md5: | ||||
|                     # * A regular expression; start the string with re: | ||||
|                     # * Any Python type (for example int or float) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             def _real_extract(self, url): | ||||
|                 mobj = re.match(self._VALID_URL, url) | ||||
|                 video_id = mobj.group('id') | ||||
|  | ||||
|                 # TODO more code goes here, for example ... | ||||
|                 webpage = self._download_webpage(url, video_id) | ||||
|                 title = self._html_search_regex(r'<h1>(.*?)</h1>', webpage, 'title') | ||||
|  | ||||
|                 return { | ||||
|                     'id': video_id, | ||||
|                     'title': title, | ||||
|                     # TODO more properties (see youtube_dl/extractor/common.py) | ||||
|                 } | ||||
|  | ||||
|  | ||||
| 5. Add an import in [`youtube_dl/extractor/__init__.py`](https://github.com/rg3/youtube-dl/blob/master/youtube_dl/extractor/__init__.py). | ||||
| 6. Run `python test/test_download.py TestDownload.test_YourExtractor`. This *should fail* at first, but you can continually re-run it until you're done. | ||||
| 7. Have a look at [`youtube_dl/common/extractor/common.py`](https://github.com/rg3/youtube-dl/blob/master/youtube_dl/extractor/common.py) for possible helper methods and a [detailed description of what your extractor should return](https://github.com/rg3/youtube-dl/blob/master/youtube_dl/extractor/common.py#L38). Add tests and code for as many as you want. | ||||
| 8. If you can, check the code with [pyflakes](https://pypi.python.org/pypi/pyflakes) (a good idea) and [pep8](https://pypi.python.org/pypi/pep8) (optional, ignore E501). | ||||
| 9. When the tests pass, [add](https://www.kernel.org/pub/software/scm/git/docs/git-add.html) the new files and [commit](https://www.kernel.org/pub/software/scm/git/docs/git-commit.html) them and [push](https://www.kernel.org/pub/software/scm/git/docs/git-push.html) the result, like this: | ||||
|  | ||||
|         $ git add youtube_dl/extractor/__init__.py | ||||
|         $ git add youtube_dl/extractor/yourextractor.py | ||||
|         $ git commit -m '[yourextractor] Add new extractor' | ||||
|         $ git push origin yourextractor | ||||
|  | ||||
| 10. Finally, [create a pull request](https://help.github.com/articles/creating-a-pull-request). We'll then review and merge it. | ||||
|  | ||||
| In any case, thank you very much for your contributions! | ||||
|  | ||||
| # BUGS | ||||
|  | ||||
| @@ -390,7 +471,7 @@ 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. | ||||
| 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? | ||||
|  | ||||
|   | ||||
| @@ -15,7 +15,7 @@ header = oldreadme[:oldreadme.index('# OPTIONS')] | ||||
| footer = oldreadme[oldreadme.index('# CONFIGURATION'):] | ||||
|  | ||||
| options = helptext[helptext.index('  General Options:') + 19:] | ||||
| options = re.sub(r'^  (\w.+)$', r'## \1', options, flags=re.M) | ||||
| options = re.sub(r'(?m)^  (\w.+)$', r'## \1', options) | ||||
| options = '# OPTIONS\n' + options + '\n' | ||||
|  | ||||
| with io.open(README_FILE, 'w', encoding='utf-8') as f: | ||||
|   | ||||
							
								
								
									
										20
									
								
								devscripts/prepare_manpage.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								devscripts/prepare_manpage.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
|  | ||||
| import io | ||||
| import os.path | ||||
| import sys | ||||
| import re | ||||
|  | ||||
| ROOT_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) | ||||
| README_FILE = os.path.join(ROOT_DIR, 'README.md') | ||||
|  | ||||
| with io.open(README_FILE, encoding='utf-8') as f: | ||||
|     readme = f.read() | ||||
|  | ||||
| PREFIX = '%YOUTUBE-DL(1)\n\n# NAME\n' | ||||
| readme = re.sub(r'(?s)# INSTALLATION.*?(?=# DESCRIPTION)', '', readme) | ||||
| readme = PREFIX + readme | ||||
|  | ||||
| if sys.version_info < (3, 0): | ||||
|     print(readme.encode('utf-8')) | ||||
| else: | ||||
|     print(readme) | ||||
| @@ -22,6 +22,12 @@ fi | ||||
|  | ||||
| if [ -z "$1" ]; then echo "ERROR: specify version number like this: $0 1994.09.06"; exit 1; fi | ||||
| version="$1" | ||||
| major_version=$(echo "$version" | sed -n 's#^\([0-9]*\.[0-9]*\.[0-9]*\).*#\1#p') | ||||
| if test "$major_version" '!=' "$(date '+%Y.%m.%d')"; then | ||||
|     echo "$version does not start with today's date!" | ||||
|     exit 1 | ||||
| fi | ||||
|  | ||||
| if [ ! -z "`git tag | grep "$version"`" ]; then echo 'ERROR: version already present'; exit 1; fi | ||||
| if [ ! -z "`git status --porcelain | grep -v CHANGELOG`" ]; then echo 'ERROR: the working directory is not clean; commit or stash changes'; exit 1; fi | ||||
| useless_files=$(find youtube_dl -type f -not -name '*.py') | ||||
| @@ -39,9 +45,9 @@ fi | ||||
| /bin/echo -e "\n### Changing version in version.py..." | ||||
| sed -i "s/__version__ = '.*'/__version__ = '$version'/" youtube_dl/version.py | ||||
|  | ||||
| /bin/echo -e "\n### Committing CHANGELOG README.md and youtube_dl/version.py..." | ||||
| /bin/echo -e "\n### Committing README.md and youtube_dl/version.py..." | ||||
| make README.md | ||||
| git add CHANGELOG README.md youtube_dl/version.py | ||||
| git add README.md youtube_dl/version.py | ||||
| git commit -m "release $version" | ||||
|  | ||||
| /bin/echo -e "\n### Now tagging, signing and pushing..." | ||||
|   | ||||
							
								
								
									
										1
									
								
								docs/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								docs/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| _build/ | ||||
							
								
								
									
										177
									
								
								docs/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										177
									
								
								docs/Makefile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,177 @@ | ||||
| # Makefile for Sphinx documentation | ||||
| # | ||||
|  | ||||
| # You can set these variables from the command line. | ||||
| SPHINXOPTS    = | ||||
| SPHINXBUILD   = sphinx-build | ||||
| PAPER         = | ||||
| BUILDDIR      = _build | ||||
|  | ||||
| # User-friendly check for sphinx-build | ||||
| ifeq ($(shell which $(SPHINXBUILD) >/dev/null 2>&1; echo $$?), 1) | ||||
| $(error The '$(SPHINXBUILD)' command was not found. Make sure you have Sphinx installed, then set the SPHINXBUILD environment variable to point to the full path of the '$(SPHINXBUILD)' executable. Alternatively you can add the directory with the executable to your PATH. If you don't have Sphinx installed, grab it from http://sphinx-doc.org/) | ||||
| endif | ||||
|  | ||||
| # Internal variables. | ||||
| PAPEROPT_a4     = -D latex_paper_size=a4 | ||||
| PAPEROPT_letter = -D latex_paper_size=letter | ||||
| ALLSPHINXOPTS   = -d $(BUILDDIR)/doctrees $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) . | ||||
| # the i18n builder cannot share the environment and doctrees with the others | ||||
| I18NSPHINXOPTS  = $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) . | ||||
|  | ||||
| .PHONY: help clean html dirhtml singlehtml pickle json htmlhelp qthelp devhelp epub latex latexpdf text man changes linkcheck doctest gettext | ||||
|  | ||||
| help: | ||||
| 	@echo "Please use \`make <target>' where <target> is one of" | ||||
| 	@echo "  html       to make standalone HTML files" | ||||
| 	@echo "  dirhtml    to make HTML files named index.html in directories" | ||||
| 	@echo "  singlehtml to make a single large HTML file" | ||||
| 	@echo "  pickle     to make pickle files" | ||||
| 	@echo "  json       to make JSON files" | ||||
| 	@echo "  htmlhelp   to make HTML files and a HTML help project" | ||||
| 	@echo "  qthelp     to make HTML files and a qthelp project" | ||||
| 	@echo "  devhelp    to make HTML files and a Devhelp project" | ||||
| 	@echo "  epub       to make an epub" | ||||
| 	@echo "  latex      to make LaTeX files, you can set PAPER=a4 or PAPER=letter" | ||||
| 	@echo "  latexpdf   to make LaTeX files and run them through pdflatex" | ||||
| 	@echo "  latexpdfja to make LaTeX files and run them through platex/dvipdfmx" | ||||
| 	@echo "  text       to make text files" | ||||
| 	@echo "  man        to make manual pages" | ||||
| 	@echo "  texinfo    to make Texinfo files" | ||||
| 	@echo "  info       to make Texinfo files and run them through makeinfo" | ||||
| 	@echo "  gettext    to make PO message catalogs" | ||||
| 	@echo "  changes    to make an overview of all changed/added/deprecated items" | ||||
| 	@echo "  xml        to make Docutils-native XML files" | ||||
| 	@echo "  pseudoxml  to make pseudoxml-XML files for display purposes" | ||||
| 	@echo "  linkcheck  to check all external links for integrity" | ||||
| 	@echo "  doctest    to run all doctests embedded in the documentation (if enabled)" | ||||
|  | ||||
| clean: | ||||
| 	rm -rf $(BUILDDIR)/* | ||||
|  | ||||
| html: | ||||
| 	$(SPHINXBUILD) -b html $(ALLSPHINXOPTS) $(BUILDDIR)/html | ||||
| 	@echo | ||||
| 	@echo "Build finished. The HTML pages are in $(BUILDDIR)/html." | ||||
|  | ||||
| dirhtml: | ||||
| 	$(SPHINXBUILD) -b dirhtml $(ALLSPHINXOPTS) $(BUILDDIR)/dirhtml | ||||
| 	@echo | ||||
| 	@echo "Build finished. The HTML pages are in $(BUILDDIR)/dirhtml." | ||||
|  | ||||
| singlehtml: | ||||
| 	$(SPHINXBUILD) -b singlehtml $(ALLSPHINXOPTS) $(BUILDDIR)/singlehtml | ||||
| 	@echo | ||||
| 	@echo "Build finished. The HTML page is in $(BUILDDIR)/singlehtml." | ||||
|  | ||||
| pickle: | ||||
| 	$(SPHINXBUILD) -b pickle $(ALLSPHINXOPTS) $(BUILDDIR)/pickle | ||||
| 	@echo | ||||
| 	@echo "Build finished; now you can process the pickle files." | ||||
|  | ||||
| json: | ||||
| 	$(SPHINXBUILD) -b json $(ALLSPHINXOPTS) $(BUILDDIR)/json | ||||
| 	@echo | ||||
| 	@echo "Build finished; now you can process the JSON files." | ||||
|  | ||||
| htmlhelp: | ||||
| 	$(SPHINXBUILD) -b htmlhelp $(ALLSPHINXOPTS) $(BUILDDIR)/htmlhelp | ||||
| 	@echo | ||||
| 	@echo "Build finished; now you can run HTML Help Workshop with the" \ | ||||
| 	      ".hhp project file in $(BUILDDIR)/htmlhelp." | ||||
|  | ||||
| qthelp: | ||||
| 	$(SPHINXBUILD) -b qthelp $(ALLSPHINXOPTS) $(BUILDDIR)/qthelp | ||||
| 	@echo | ||||
| 	@echo "Build finished; now you can run "qcollectiongenerator" with the" \ | ||||
| 	      ".qhcp project file in $(BUILDDIR)/qthelp, like this:" | ||||
| 	@echo "# qcollectiongenerator $(BUILDDIR)/qthelp/youtube-dl.qhcp" | ||||
| 	@echo "To view the help file:" | ||||
| 	@echo "# assistant -collectionFile $(BUILDDIR)/qthelp/youtube-dl.qhc" | ||||
|  | ||||
| devhelp: | ||||
| 	$(SPHINXBUILD) -b devhelp $(ALLSPHINXOPTS) $(BUILDDIR)/devhelp | ||||
| 	@echo | ||||
| 	@echo "Build finished." | ||||
| 	@echo "To view the help file:" | ||||
| 	@echo "# mkdir -p $$HOME/.local/share/devhelp/youtube-dl" | ||||
| 	@echo "# ln -s $(BUILDDIR)/devhelp $$HOME/.local/share/devhelp/youtube-dl" | ||||
| 	@echo "# devhelp" | ||||
|  | ||||
| epub: | ||||
| 	$(SPHINXBUILD) -b epub $(ALLSPHINXOPTS) $(BUILDDIR)/epub | ||||
| 	@echo | ||||
| 	@echo "Build finished. The epub file is in $(BUILDDIR)/epub." | ||||
|  | ||||
| latex: | ||||
| 	$(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex | ||||
| 	@echo | ||||
| 	@echo "Build finished; the LaTeX files are in $(BUILDDIR)/latex." | ||||
| 	@echo "Run \`make' in that directory to run these through (pdf)latex" \ | ||||
| 	      "(use \`make latexpdf' here to do that automatically)." | ||||
|  | ||||
| latexpdf: | ||||
| 	$(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex | ||||
| 	@echo "Running LaTeX files through pdflatex..." | ||||
| 	$(MAKE) -C $(BUILDDIR)/latex all-pdf | ||||
| 	@echo "pdflatex finished; the PDF files are in $(BUILDDIR)/latex." | ||||
|  | ||||
| latexpdfja: | ||||
| 	$(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex | ||||
| 	@echo "Running LaTeX files through platex and dvipdfmx..." | ||||
| 	$(MAKE) -C $(BUILDDIR)/latex all-pdf-ja | ||||
| 	@echo "pdflatex finished; the PDF files are in $(BUILDDIR)/latex." | ||||
|  | ||||
| text: | ||||
| 	$(SPHINXBUILD) -b text $(ALLSPHINXOPTS) $(BUILDDIR)/text | ||||
| 	@echo | ||||
| 	@echo "Build finished. The text files are in $(BUILDDIR)/text." | ||||
|  | ||||
| man: | ||||
| 	$(SPHINXBUILD) -b man $(ALLSPHINXOPTS) $(BUILDDIR)/man | ||||
| 	@echo | ||||
| 	@echo "Build finished. The manual pages are in $(BUILDDIR)/man." | ||||
|  | ||||
| texinfo: | ||||
| 	$(SPHINXBUILD) -b texinfo $(ALLSPHINXOPTS) $(BUILDDIR)/texinfo | ||||
| 	@echo | ||||
| 	@echo "Build finished. The Texinfo files are in $(BUILDDIR)/texinfo." | ||||
| 	@echo "Run \`make' in that directory to run these through makeinfo" \ | ||||
| 	      "(use \`make info' here to do that automatically)." | ||||
|  | ||||
| info: | ||||
| 	$(SPHINXBUILD) -b texinfo $(ALLSPHINXOPTS) $(BUILDDIR)/texinfo | ||||
| 	@echo "Running Texinfo files through makeinfo..." | ||||
| 	make -C $(BUILDDIR)/texinfo info | ||||
| 	@echo "makeinfo finished; the Info files are in $(BUILDDIR)/texinfo." | ||||
|  | ||||
| gettext: | ||||
| 	$(SPHINXBUILD) -b gettext $(I18NSPHINXOPTS) $(BUILDDIR)/locale | ||||
| 	@echo | ||||
| 	@echo "Build finished. The message catalogs are in $(BUILDDIR)/locale." | ||||
|  | ||||
| changes: | ||||
| 	$(SPHINXBUILD) -b changes $(ALLSPHINXOPTS) $(BUILDDIR)/changes | ||||
| 	@echo | ||||
| 	@echo "The overview file is in $(BUILDDIR)/changes." | ||||
|  | ||||
| linkcheck: | ||||
| 	$(SPHINXBUILD) -b linkcheck $(ALLSPHINXOPTS) $(BUILDDIR)/linkcheck | ||||
| 	@echo | ||||
| 	@echo "Link check complete; look for any errors in the above output " \ | ||||
| 	      "or in $(BUILDDIR)/linkcheck/output.txt." | ||||
|  | ||||
| doctest: | ||||
| 	$(SPHINXBUILD) -b doctest $(ALLSPHINXOPTS) $(BUILDDIR)/doctest | ||||
| 	@echo "Testing of doctests in the sources finished, look at the " \ | ||||
| 	      "results in $(BUILDDIR)/doctest/output.txt." | ||||
|  | ||||
| xml: | ||||
| 	$(SPHINXBUILD) -b xml $(ALLSPHINXOPTS) $(BUILDDIR)/xml | ||||
| 	@echo | ||||
| 	@echo "Build finished. The XML files are in $(BUILDDIR)/xml." | ||||
|  | ||||
| pseudoxml: | ||||
| 	$(SPHINXBUILD) -b pseudoxml $(ALLSPHINXOPTS) $(BUILDDIR)/pseudoxml | ||||
| 	@echo | ||||
| 	@echo "Build finished. The pseudo-XML files are in $(BUILDDIR)/pseudoxml." | ||||
							
								
								
									
										71
									
								
								docs/conf.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								docs/conf.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| # | ||||
| # youtube-dl documentation build configuration file, created by | ||||
| # sphinx-quickstart on Fri Mar 14 21:05:43 2014. | ||||
| # | ||||
| # This file is execfile()d with the current directory set to its | ||||
| # containing dir. | ||||
| # | ||||
| # Note that not all possible configuration values are present in this | ||||
| # autogenerated file. | ||||
| # | ||||
| # All configuration values have a default; values that are commented out | ||||
| # serve to show the default. | ||||
|  | ||||
| import sys | ||||
| import os | ||||
| # Allows to import youtube_dl | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| # -- General configuration ------------------------------------------------ | ||||
|  | ||||
| # Add any Sphinx extension module names here, as strings. They can be | ||||
| # extensions coming with Sphinx (named 'sphinx.ext.*') or your custom | ||||
| # ones. | ||||
| extensions = [ | ||||
|     'sphinx.ext.autodoc', | ||||
| ] | ||||
|  | ||||
| # Add any paths that contain templates here, relative to this directory. | ||||
| templates_path = ['_templates'] | ||||
|  | ||||
| # The suffix of source filenames. | ||||
| source_suffix = '.rst' | ||||
|  | ||||
| # The master toctree document. | ||||
| master_doc = 'index' | ||||
|  | ||||
| # General information about the project. | ||||
| project = u'youtube-dl' | ||||
| copyright = u'2014, Ricardo Garcia Gonzalez' | ||||
|  | ||||
| # The version info for the project you're documenting, acts as replacement for | ||||
| # |version| and |release|, also used in various other places throughout the | ||||
| # built documents. | ||||
| # | ||||
| # The short X.Y version. | ||||
| import youtube_dl | ||||
| version = youtube_dl.__version__ | ||||
| # The full version, including alpha/beta/rc tags. | ||||
| release = version | ||||
|  | ||||
| # List of patterns, relative to source directory, that match files and | ||||
| # directories to ignore when looking for source files. | ||||
| exclude_patterns = ['_build'] | ||||
|  | ||||
| # The name of the Pygments (syntax highlighting) style to use. | ||||
| pygments_style = 'sphinx' | ||||
|  | ||||
| # -- Options for HTML output ---------------------------------------------- | ||||
|  | ||||
| # The theme to use for HTML and HTML Help pages.  See the documentation for | ||||
| # a list of builtin themes. | ||||
| html_theme = 'default' | ||||
|  | ||||
| # Add any paths that contain custom static files (such as style sheets) here, | ||||
| # relative to this directory. They are copied after the builtin static files, | ||||
| # so a file named "default.css" will overwrite the builtin "default.css". | ||||
| html_static_path = ['_static'] | ||||
|  | ||||
| # Output file base name for HTML help builder. | ||||
| htmlhelp_basename = 'youtube-dldoc' | ||||
							
								
								
									
										23
									
								
								docs/index.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								docs/index.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| Welcome to youtube-dl's documentation! | ||||
| ====================================== | ||||
|  | ||||
| *youtube-dl* is a command-line program to download videos from YouTube.com and more sites. | ||||
| It can also be used in Python code. | ||||
|  | ||||
| Developer guide | ||||
| --------------- | ||||
|  | ||||
| This section contains information for using *youtube-dl* from Python programs. | ||||
|  | ||||
| .. toctree:: | ||||
|     :maxdepth: 2 | ||||
|  | ||||
|     module_guide | ||||
|  | ||||
| Indices and tables | ||||
| ================== | ||||
|  | ||||
| * :ref:`genindex` | ||||
| * :ref:`modindex` | ||||
| * :ref:`search` | ||||
|  | ||||
							
								
								
									
										67
									
								
								docs/module_guide.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								docs/module_guide.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | ||||
| Using the ``youtube_dl`` module | ||||
| =============================== | ||||
|  | ||||
| When using the ``youtube_dl`` module, you start by creating an instance of :class:`YoutubeDL` and adding all the available extractors: | ||||
|  | ||||
| .. code-block:: python | ||||
|  | ||||
|     >>> from youtube_dl import YoutubeDL | ||||
|     >>> ydl = YoutubeDL() | ||||
|     >>> ydl.add_default_info_extractors() | ||||
|  | ||||
| Extracting video information | ||||
| ---------------------------- | ||||
|  | ||||
| You use the :meth:`YoutubeDL.extract_info` method for getting the video information, which returns a dictionary: | ||||
|  | ||||
| .. code-block:: python | ||||
|  | ||||
|     >>> info = ydl.extract_info('http://www.youtube.com/watch?v=BaW_jenozKc', download=False) | ||||
|     [youtube] Setting language | ||||
|     [youtube] BaW_jenozKc: Downloading webpage | ||||
|     [youtube] BaW_jenozKc: Downloading video info webpage | ||||
|     [youtube] BaW_jenozKc: Extracting video information | ||||
|     >>> info['title'] | ||||
|     'youtube-dl test video "\'/\\ä↭𝕐' | ||||
|     >>> info['height'], info['width'] | ||||
|     (720, 1280) | ||||
|  | ||||
| If you want to download or play the video you can get its url: | ||||
|  | ||||
| .. code-block:: python | ||||
|  | ||||
|     >>> info['url'] | ||||
|     'https://...' | ||||
|  | ||||
| Extracting playlist information | ||||
| ------------------------------- | ||||
|  | ||||
| The playlist information is extracted in a similar way, but the dictionary is a bit different: | ||||
|  | ||||
| .. code-block:: python | ||||
|  | ||||
|     >>> playlist = ydl.extract_info('http://www.ted.com/playlists/13/open_source_open_world', download=False) | ||||
|     [TED] open_source_open_world: Downloading playlist webpage | ||||
|     ... | ||||
|     >>> playlist['title'] | ||||
|     'Open-source, open world' | ||||
|  | ||||
|  | ||||
|  | ||||
| You can access the videos in the playlist with the ``entries`` field: | ||||
|  | ||||
| .. code-block:: python | ||||
|  | ||||
|     >>> for video in playlist['entries']: | ||||
|     ...     print('Video #%d: %s' % (video['playlist_index'], video['title'])) | ||||
|  | ||||
|     Video #1: How Arduino is open-sourcing imagination | ||||
|     Video #2: The year open data went worldwide | ||||
|     Video #3: Massive-scale online collaboration | ||||
|     Video #4: The art of asking | ||||
|     Video #5: How cognitive surplus will change the world | ||||
|     Video #6: The birth of Wikipedia | ||||
|     Video #7: Coding a better government | ||||
|     Video #8: The era of open innovation | ||||
|     Video #9: The currency of the new economy is trust | ||||
|  | ||||
| @@ -74,13 +74,19 @@ class FakeYDL(YoutubeDL): | ||||
|             old_report_warning(message) | ||||
|         self.report_warning = types.MethodType(report_warning, self) | ||||
|  | ||||
| def gettestcases(): | ||||
|  | ||||
| def gettestcases(include_onlymatching=False): | ||||
|     for ie in youtube_dl.extractor.gen_extractors(): | ||||
|         t = getattr(ie, '_TEST', None) | ||||
|         if t: | ||||
|             t['name'] = type(ie).__name__[:-len('IE')] | ||||
|             yield t | ||||
|         for t in getattr(ie, '_TESTS', []): | ||||
|             assert not hasattr(ie, '_TESTS'), \ | ||||
|                 '%s has _TEST and _TESTS' % type(ie).__name__ | ||||
|             tests = [t] | ||||
|         else: | ||||
|             tests = getattr(ie, '_TESTS', []) | ||||
|         for t in tests: | ||||
|             if not include_onlymatching and t.get('only_matching', False): | ||||
|                 continue | ||||
|             t['name'] = type(ie).__name__[:-len('IE')] | ||||
|             yield t | ||||
|  | ||||
| @@ -101,7 +107,7 @@ def expect_info_dict(self, expected_dict, got_dict): | ||||
|         elif isinstance(expected, type): | ||||
|             got = got_dict.get(info_field) | ||||
|             self.assertTrue(isinstance(got, expected), | ||||
|                 u'Expected type %r, but got value %r of type %r' % (expected, got, type(got))) | ||||
|                 u'Expected type %r for field %s, but got value %r of type %r' % (expected, info_field, got, type(got))) | ||||
|         else: | ||||
|             if isinstance(expected, compat_str) and expected.startswith('md5:'): | ||||
|                 got = 'md5:' + md5(got_dict.get(info_field)) | ||||
| @@ -128,3 +134,17 @@ def expect_info_dict(self, expected_dict, got_dict): | ||||
|             missing_keys, | ||||
|             'Missing keys in test definition: %s' % ( | ||||
|                 ', '.join(sorted(missing_keys)))) | ||||
|  | ||||
|  | ||||
| def assertRegexpMatches(self, text, regexp, msg=None): | ||||
|     if hasattr(self, 'assertRegexpMatches'): | ||||
|         return self.assertRegexpMatches(text, regexp, msg) | ||||
|     else: | ||||
|         m = re.match(regexp, text) | ||||
|         if not m: | ||||
|             note = 'Regexp didn\'t match: %r not found in %r' % (regexp, text) | ||||
|             if msg is None: | ||||
|                 msg = note | ||||
|             else: | ||||
|                 msg = note + ', ' + msg | ||||
|             self.assertTrue(m, msg) | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import sys | ||||
| import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from test.helper import FakeYDL | ||||
| from test.helper import FakeYDL, assertRegexpMatches | ||||
| from youtube_dl import YoutubeDL | ||||
| from youtube_dl.extractor import YoutubeIE | ||||
|  | ||||
| @@ -26,16 +26,27 @@ class YDL(FakeYDL): | ||||
|         self.msgs.append(msg) | ||||
|  | ||||
|  | ||||
| def _make_result(formats, **kwargs): | ||||
|     res = { | ||||
|         'formats': formats, | ||||
|         'id': 'testid', | ||||
|         'title': 'testttitle', | ||||
|         'extractor': 'testex', | ||||
|     } | ||||
|     res.update(**kwargs) | ||||
|     return res | ||||
|  | ||||
|  | ||||
| class TestFormatSelection(unittest.TestCase): | ||||
|     def test_prefer_free_formats(self): | ||||
|         # Same resolution => download webm | ||||
|         ydl = YDL() | ||||
|         ydl.params['prefer_free_formats'] = True | ||||
|         formats = [ | ||||
|             {'ext': 'webm', 'height': 460}, | ||||
|             {'ext': 'mp4',  'height': 460}, | ||||
|             {'ext': 'webm', 'height': 460, 'url': 'x'}, | ||||
|             {'ext': 'mp4', 'height': 460, 'url': 'y'}, | ||||
|         ] | ||||
|         info_dict = {'formats': formats, 'extractor': 'test'} | ||||
|         info_dict = _make_result(formats) | ||||
|         yie = YoutubeIE(ydl) | ||||
|         yie._sort_formats(info_dict['formats']) | ||||
|         ydl.process_ie_result(info_dict) | ||||
| @@ -46,8 +57,8 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         ydl = YDL() | ||||
|         ydl.params['prefer_free_formats'] = True | ||||
|         formats = [ | ||||
|             {'ext': 'webm', 'height': 720}, | ||||
|             {'ext': 'mp4', 'height': 1080}, | ||||
|             {'ext': 'webm', 'height': 720, 'url': 'a'}, | ||||
|             {'ext': 'mp4', 'height': 1080, 'url': 'b'}, | ||||
|         ] | ||||
|         info_dict['formats'] = formats | ||||
|         yie = YoutubeIE(ydl) | ||||
| @@ -56,13 +67,13 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['ext'], 'mp4') | ||||
|  | ||||
|         # No prefer_free_formats => prefer mp4 and flv for greater compatibilty | ||||
|         # No prefer_free_formats => prefer mp4 and flv for greater compatibility | ||||
|         ydl = YDL() | ||||
|         ydl.params['prefer_free_formats'] = False | ||||
|         formats = [ | ||||
|             {'ext': 'webm', 'height': 720}, | ||||
|             {'ext': 'mp4', 'height': 720}, | ||||
|             {'ext': 'flv', 'height': 720}, | ||||
|             {'ext': 'webm', 'height': 720, 'url': '_'}, | ||||
|             {'ext': 'mp4', 'height': 720, 'url': '_'}, | ||||
|             {'ext': 'flv', 'height': 720, 'url': '_'}, | ||||
|         ] | ||||
|         info_dict['formats'] = formats | ||||
|         yie = YoutubeIE(ydl) | ||||
| @@ -74,8 +85,8 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         ydl = YDL() | ||||
|         ydl.params['prefer_free_formats'] = False | ||||
|         formats = [ | ||||
|             {'ext': 'flv', 'height': 720}, | ||||
|             {'ext': 'webm', 'height': 720}, | ||||
|             {'ext': 'flv', 'height': 720, 'url': '_'}, | ||||
|             {'ext': 'webm', 'height': 720, 'url': '_'}, | ||||
|         ] | ||||
|         info_dict['formats'] = formats | ||||
|         yie = YoutubeIE(ydl) | ||||
| @@ -91,8 +102,7 @@ class TestFormatSelection(unittest.TestCase): | ||||
|             {'format_id': 'great', 'url': 'http://example.com/great', 'preference': 3}, | ||||
|             {'format_id': 'excellent', 'url': 'http://example.com/exc', 'preference': 4}, | ||||
|         ] | ||||
|         info_dict = { | ||||
|             'formats': formats, 'extractor': 'test', 'id': 'testvid'} | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
|         ydl = YDL() | ||||
|         ydl.process_ie_result(info_dict) | ||||
| @@ -120,12 +130,12 @@ class TestFormatSelection(unittest.TestCase): | ||||
|  | ||||
|     def test_format_selection(self): | ||||
|         formats = [ | ||||
|             {'format_id': '35', 'ext': 'mp4', 'preference': 1}, | ||||
|             {'format_id': '45', 'ext': 'webm', 'preference': 2}, | ||||
|             {'format_id': '47', 'ext': 'webm', 'preference': 3}, | ||||
|             {'format_id': '2', 'ext': 'flv', 'preference': 4}, | ||||
|             {'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': '_'}, | ||||
|         ] | ||||
|         info_dict = {'formats': formats, 'extractor': 'test'} | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
|         ydl = YDL({'format': '20/47'}) | ||||
|         ydl.process_ie_result(info_dict.copy()) | ||||
| @@ -154,12 +164,12 @@ class TestFormatSelection(unittest.TestCase): | ||||
|  | ||||
|     def test_format_selection_audio(self): | ||||
|         formats = [ | ||||
|             {'format_id': 'audio-low', 'ext': 'webm', 'preference': 1, 'vcodec': 'none'}, | ||||
|             {'format_id': 'audio-mid', 'ext': 'webm', 'preference': 2, 'vcodec': 'none'}, | ||||
|             {'format_id': 'audio-high', 'ext': 'flv', 'preference': 3, 'vcodec': 'none'}, | ||||
|             {'format_id': 'vid', 'ext': 'mp4', 'preference': 4}, | ||||
|             {'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': '_'}, | ||||
|         ] | ||||
|         info_dict = {'formats': formats, 'extractor': 'test'} | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
|         ydl = YDL({'format': 'bestaudio'}) | ||||
|         ydl.process_ie_result(info_dict.copy()) | ||||
| @@ -172,10 +182,10 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         self.assertEqual(downloaded['format_id'], 'audio-low') | ||||
|  | ||||
|         formats = [ | ||||
|             {'format_id': 'vid-low', 'ext': 'mp4', 'preference': 1}, | ||||
|             {'format_id': 'vid-high', 'ext': 'mp4', 'preference': 2}, | ||||
|             {'format_id': 'vid-low', 'ext': 'mp4', 'preference': 1, 'url': '_'}, | ||||
|             {'format_id': 'vid-high', 'ext': 'mp4', 'preference': 2, 'url': '_'}, | ||||
|         ] | ||||
|         info_dict = {'formats': formats, 'extractor': 'test'} | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
|         ydl = YDL({'format': 'bestaudio/worstaudio/best'}) | ||||
|         ydl.process_ie_result(info_dict.copy()) | ||||
| @@ -184,11 +194,11 @@ class TestFormatSelection(unittest.TestCase): | ||||
|  | ||||
|     def test_format_selection_video(self): | ||||
|         formats = [ | ||||
|             {'format_id': 'dash-video-low', 'ext': 'mp4', 'preference': 1, 'acodec': 'none'}, | ||||
|             {'format_id': 'dash-video-high', 'ext': 'mp4', 'preference': 2, 'acodec': 'none'}, | ||||
|             {'format_id': 'vid', 'ext': 'mp4', 'preference': 3}, | ||||
|             {'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': '_'}, | ||||
|         ] | ||||
|         info_dict = {'formats': formats, 'extractor': 'test'} | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
|         ydl = YDL({'format': 'bestvideo'}) | ||||
|         ydl.process_ie_result(info_dict.copy()) | ||||
| @@ -217,10 +227,12 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         for f1id, f2id in zip(order, order[1:]): | ||||
|             f1 = YoutubeIE._formats[f1id].copy() | ||||
|             f1['format_id'] = f1id | ||||
|             f1['url'] = 'url:' + f1id | ||||
|             f2 = YoutubeIE._formats[f2id].copy() | ||||
|             f2['format_id'] = f2id | ||||
|             f2['url'] = 'url:' + f2id | ||||
|  | ||||
|             info_dict = {'formats': [f1, f2], 'extractor': 'youtube'} | ||||
|             info_dict = _make_result([f1, f2], extractor='youtube') | ||||
|             ydl = YDL() | ||||
|             yie = YoutubeIE(ydl) | ||||
|             yie._sort_formats(info_dict['formats']) | ||||
| @@ -228,7 +240,7 @@ class TestFormatSelection(unittest.TestCase): | ||||
|             downloaded = ydl.downloaded_info_dicts[0] | ||||
|             self.assertEqual(downloaded['format_id'], f1id) | ||||
|  | ||||
|             info_dict = {'formats': [f2, f1], 'extractor': 'youtube'} | ||||
|             info_dict = _make_result([f2, f1], extractor='youtube') | ||||
|             ydl = YDL() | ||||
|             yie = YoutubeIE(ydl) | ||||
|             yie._sort_formats(info_dict['formats']) | ||||
| @@ -262,6 +274,12 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         # Replace missing fields with 'NA' | ||||
|         self.assertEqual(fname('%(uploader_date)s-%(id)s.%(ext)s'), 'NA-1234.mp4') | ||||
|  | ||||
|     def test_format_note(self): | ||||
|         ydl = YoutubeDL() | ||||
|         self.assertEqual(ydl._format_note({}), '') | ||||
|         assertRegexpMatches(self, ydl._format_note({ | ||||
|             'vbr': 10, | ||||
|         }), '^\s*10k$') | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -13,7 +13,7 @@ from youtube_dl import YoutubeDL | ||||
|  | ||||
|  | ||||
| def _download_restricted(url, filename, age): | ||||
|     """ Returns true iff the file has been downloaded """ | ||||
|     """ Returns true if the file has been downloaded """ | ||||
|  | ||||
|     params = { | ||||
|         'age_limit': age, | ||||
|   | ||||
| @@ -49,6 +49,7 @@ class TestAllURLsMatching(unittest.TestCase): | ||||
|         self.assertMatch('http://youtu.be/BaW_jenozKc', ['youtube']) | ||||
|         self.assertMatch('http://www.youtube.com/v/BaW_jenozKc', ['youtube']) | ||||
|         self.assertMatch('https://youtube.googleapis.com/v/BaW_jenozKc', ['youtube']) | ||||
|         self.assertMatch('http://www.cleanvideosearch.com/media/action/yt/watch?videoId=8v_4O44sfjM', ['youtube']) | ||||
|  | ||||
|     def test_youtube_channel_matching(self): | ||||
|         assertChannel = lambda url: self.assertMatch(url, ['youtube:channel']) | ||||
| @@ -76,20 +77,20 @@ class TestAllURLsMatching(unittest.TestCase): | ||||
|         self.assertMatch('https://www.youtube.com/results?baz=bar&search_query=youtube-dl+test+video&filters=video&lclk=video', ['youtube:search_url']) | ||||
|  | ||||
|     def test_justin_tv_channelid_matching(self): | ||||
|         self.assertTrue(JustinTVIE.suitable(u"justin.tv/vanillatv")) | ||||
|         self.assertTrue(JustinTVIE.suitable(u"twitch.tv/vanillatv")) | ||||
|         self.assertTrue(JustinTVIE.suitable(u"www.justin.tv/vanillatv")) | ||||
|         self.assertTrue(JustinTVIE.suitable(u"www.twitch.tv/vanillatv")) | ||||
|         self.assertTrue(JustinTVIE.suitable(u"http://www.justin.tv/vanillatv")) | ||||
|         self.assertTrue(JustinTVIE.suitable(u"http://www.twitch.tv/vanillatv")) | ||||
|         self.assertTrue(JustinTVIE.suitable(u"http://www.justin.tv/vanillatv/")) | ||||
|         self.assertTrue(JustinTVIE.suitable(u"http://www.twitch.tv/vanillatv/")) | ||||
|         self.assertTrue(JustinTVIE.suitable('justin.tv/vanillatv')) | ||||
|         self.assertTrue(JustinTVIE.suitable('twitch.tv/vanillatv')) | ||||
|         self.assertTrue(JustinTVIE.suitable('www.justin.tv/vanillatv')) | ||||
|         self.assertTrue(JustinTVIE.suitable('www.twitch.tv/vanillatv')) | ||||
|         self.assertTrue(JustinTVIE.suitable('http://www.justin.tv/vanillatv')) | ||||
|         self.assertTrue(JustinTVIE.suitable('http://www.twitch.tv/vanillatv')) | ||||
|         self.assertTrue(JustinTVIE.suitable('http://www.justin.tv/vanillatv/')) | ||||
|         self.assertTrue(JustinTVIE.suitable('http://www.twitch.tv/vanillatv/')) | ||||
|  | ||||
|     def test_justintv_videoid_matching(self): | ||||
|         self.assertTrue(JustinTVIE.suitable(u"http://www.twitch.tv/vanillatv/b/328087483")) | ||||
|         self.assertTrue(JustinTVIE.suitable('http://www.twitch.tv/vanillatv/b/328087483')) | ||||
|  | ||||
|     def test_justin_tv_chapterid_matching(self): | ||||
|         self.assertTrue(JustinTVIE.suitable(u"http://www.twitch.tv/tsm_theoddone/c/2349361")) | ||||
|         self.assertTrue(JustinTVIE.suitable('http://www.twitch.tv/tsm_theoddone/c/2349361')) | ||||
|  | ||||
|     def test_youtube_extract(self): | ||||
|         assertExtractId = lambda url, id: self.assertEqual(YoutubeIE.extract_id(url), id) | ||||
| @@ -105,7 +106,7 @@ class TestAllURLsMatching(unittest.TestCase): | ||||
|  | ||||
|     def test_no_duplicates(self): | ||||
|         ies = gen_extractors() | ||||
|         for tc in gettestcases(): | ||||
|         for tc in gettestcases(include_onlymatching=True): | ||||
|             url = tc['url'] | ||||
|             for ie in ies: | ||||
|                 if type(ie).__name__ in ('GenericIE', tc['name'] + 'IE'): | ||||
| @@ -143,5 +144,38 @@ class TestAllURLsMatching(unittest.TestCase): | ||||
|         self.assertMatch('http://video.pbs.org/viralplayer/2365173446/', ['PBS']) | ||||
|         self.assertMatch('http://video.pbs.org/widget/partnerplayer/980042464/', ['PBS']) | ||||
|  | ||||
|     def test_ComedyCentralShows(self): | ||||
|         self.assertMatch( | ||||
|             'http://thedailyshow.cc.com/extended-interviews/xm3fnq/andrew-napolitano-extended-interview', | ||||
|             ['ComedyCentralShows']) | ||||
|         self.assertMatch( | ||||
|             'http://thecolbertreport.cc.com/videos/29w6fx/-realhumanpraise-for-fox-news', | ||||
|             ['ComedyCentralShows']) | ||||
|         self.assertMatch( | ||||
|             'http://thecolbertreport.cc.com/videos/gh6urb/neil-degrasse-tyson-pt--1?xrs=eml_col_031114', | ||||
|             ['ComedyCentralShows']) | ||||
|         self.assertMatch( | ||||
|             'http://thedailyshow.cc.com/guests/michael-lewis/3efna8/exclusive---michael-lewis-extended-interview-pt--3', | ||||
|             ['ComedyCentralShows']) | ||||
|         self.assertMatch( | ||||
|             'http://thedailyshow.cc.com/episodes/sy7yv0/april-8--2014---denis-leary', | ||||
|             ['ComedyCentralShows']) | ||||
|         self.assertMatch( | ||||
|             'http://thecolbertreport.cc.com/episodes/8ase07/april-8--2014---jane-goodall', | ||||
|             ['ComedyCentralShows']) | ||||
|         self.assertMatch( | ||||
|             'http://thedailyshow.cc.com/video-playlists/npde3s/the-daily-show-19088-highlights', | ||||
|             ['ComedyCentralShows']) | ||||
|         self.assertMatch( | ||||
|             'http://thedailyshow.cc.com/special-editions/2l8fdb/special-edition---a-look-back-at-food', | ||||
|             ['ComedyCentralShows']) | ||||
|  | ||||
|     def test_yahoo_https(self): | ||||
|         # https://github.com/rg3/youtube-dl/issues/2701 | ||||
|         self.assertMatch( | ||||
|             'https://screen.yahoo.com/smartwatches-latest-wearable-gadgets-163745379-cbs.html', | ||||
|             ['Yahoo']) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -10,6 +10,7 @@ import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from test.helper import ( | ||||
|     assertRegexpMatches, | ||||
|     expect_info_dict, | ||||
|     FakeYDL, | ||||
| ) | ||||
| @@ -22,10 +23,14 @@ from youtube_dl.extractor import ( | ||||
|     VimeoUserIE, | ||||
|     VimeoAlbumIE, | ||||
|     VimeoGroupsIE, | ||||
|     VineUserIE, | ||||
|     UstreamChannelIE, | ||||
|     SoundcloudSetIE, | ||||
|     SoundcloudUserIE, | ||||
|     SoundcloudPlaylistIE, | ||||
|     TeacherTubeClassroomIE, | ||||
|     LivestreamIE, | ||||
|     LivestreamOriginalIE, | ||||
|     NHLVideocenterIE, | ||||
|     BambuserChannelIE, | ||||
|     BandcampAlbumIE, | ||||
| @@ -36,12 +41,15 @@ from youtube_dl.extractor import ( | ||||
|     KhanAcademyIE, | ||||
|     EveryonesMixtapeIE, | ||||
|     RutubeChannelIE, | ||||
|     RutubePersonIE, | ||||
|     GoogleSearchIE, | ||||
|     GenericIE, | ||||
|     TEDIE, | ||||
|     ToypicsUserIE, | ||||
|     XTubeUserIE, | ||||
|     InstagramUserIE, | ||||
|     CSpanIE, | ||||
|     AolIE, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -98,13 +106,20 @@ class TestPlaylists(unittest.TestCase): | ||||
|         self.assertEqual(result['title'], 'Rolex Awards for Enterprise') | ||||
|         self.assertTrue(len(result['entries']) > 72) | ||||
|  | ||||
|     def test_vine_user(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = VineUserIE(dl) | ||||
|         result = ie.extract('https://vine.co/Visa') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertTrue(len(result['entries']) >= 50) | ||||
|  | ||||
|     def test_ustream_channel(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = UstreamChannelIE(dl) | ||||
|         result = ie.extract('http://www.ustream.tv/channel/young-americans-for-liberty') | ||||
|         result = ie.extract('http://www.ustream.tv/channel/channeljapan') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], '5124905') | ||||
|         self.assertTrue(len(result['entries']) >= 6) | ||||
|         self.assertEqual(result['id'], '10874166') | ||||
|         self.assertTrue(len(result['entries']) >= 54) | ||||
|  | ||||
|     def test_soundcloud_set(self): | ||||
|         dl = FakeYDL() | ||||
| @@ -122,6 +137,17 @@ class TestPlaylists(unittest.TestCase): | ||||
|         self.assertEqual(result['id'], '9615865') | ||||
|         self.assertTrue(len(result['entries']) >= 12) | ||||
|  | ||||
|     def test_soundcloud_playlist(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = SoundcloudPlaylistIE(dl) | ||||
|         result = ie.extract('http://api.soundcloud.com/playlists/4110309') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], '4110309') | ||||
|         self.assertEqual(result['title'], 'TILT Brass - Bowery Poetry Club, August \'03 [Non-Site SCR 02]') | ||||
|         assertRegexpMatches( | ||||
|             self, result['description'], r'TILT Brass - Bowery Poetry Club') | ||||
|         self.assertEqual(len(result['entries']), 6) | ||||
|  | ||||
|     def test_livestream_event(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = LivestreamIE(dl) | ||||
| @@ -130,6 +156,14 @@ class TestPlaylists(unittest.TestCase): | ||||
|         self.assertEqual(result['title'], 'TEDCity2.0 (English)') | ||||
|         self.assertTrue(len(result['entries']) >= 4) | ||||
|  | ||||
|     def test_livestreamoriginal_folder(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = LivestreamOriginalIE(dl) | ||||
|         result = ie.extract('https://www.livestream.com/newplay/folder?dirId=a07bf706-d0e4-4e75-a747-b021d84f2fd3') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], 'a07bf706-d0e4-4e75-a747-b021d84f2fd3') | ||||
|         self.assertTrue(len(result['entries']) >= 28) | ||||
|  | ||||
|     def test_nhl_videocenter(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = NHLVideocenterIE(dl) | ||||
| @@ -186,20 +220,20 @@ class TestPlaylists(unittest.TestCase): | ||||
|     def test_ivi_compilation(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = IviCompilationIE(dl) | ||||
|         result = ie.extract('http://www.ivi.ru/watch/dezhurnyi_angel') | ||||
|         result = ie.extract('http://www.ivi.ru/watch/dvoe_iz_lartsa') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], 'dezhurnyi_angel') | ||||
|         self.assertEqual(result['title'], 'Дежурный ангел (2010 - 2012)') | ||||
|         self.assertTrue(len(result['entries']) >= 36) | ||||
|          | ||||
|         self.assertEqual(result['id'], 'dvoe_iz_lartsa') | ||||
|         self.assertEqual(result['title'], 'Двое из ларца (2006 - 2008)') | ||||
|         self.assertTrue(len(result['entries']) >= 24) | ||||
|  | ||||
|     def test_ivi_compilation_season(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = IviCompilationIE(dl) | ||||
|         result = ie.extract('http://www.ivi.ru/watch/dezhurnyi_angel/season2') | ||||
|         result = ie.extract('http://www.ivi.ru/watch/dvoe_iz_lartsa/season1') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], 'dezhurnyi_angel/season2') | ||||
|         self.assertEqual(result['title'], 'Дежурный ангел (2010 - 2012) 2 сезон') | ||||
|         self.assertTrue(len(result['entries']) >= 20) | ||||
|         self.assertEqual(result['id'], 'dvoe_iz_lartsa/season1') | ||||
|         self.assertEqual(result['title'], 'Двое из ларца (2006 - 2008) 1 сезон') | ||||
|         self.assertTrue(len(result['entries']) >= 12) | ||||
|          | ||||
|     def test_imdb_list(self): | ||||
|         dl = FakeYDL() | ||||
| @@ -232,10 +266,18 @@ class TestPlaylists(unittest.TestCase): | ||||
|     def test_rutube_channel(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = RutubeChannelIE(dl) | ||||
|         result = ie.extract('http://rutube.ru/tags/video/1409') | ||||
|         result = ie.extract('http://rutube.ru/tags/video/1800/') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], '1409') | ||||
|         self.assertTrue(len(result['entries']) >= 34) | ||||
|         self.assertEqual(result['id'], '1800') | ||||
|         self.assertTrue(len(result['entries']) >= 68) | ||||
|  | ||||
|     def test_rutube_person(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = RutubePersonIE(dl) | ||||
|         result = ie.extract('http://rutube.ru/video/person/313878/') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], '313878') | ||||
|         self.assertTrue(len(result['entries']) >= 37) | ||||
|  | ||||
|     def test_multiple_brightcove_videos(self): | ||||
|         # https://github.com/rg3/youtube-dl/issues/2283 | ||||
| @@ -309,9 +351,41 @@ class TestPlaylists(unittest.TestCase): | ||||
|             'thumbnail': 're:^https?://.*\.jpg', | ||||
|             'uploader': 'Porsche', | ||||
|             'uploader_id': 'porsche', | ||||
|             'timestamp': 1387486713, | ||||
|             'upload_date': '20131219', | ||||
|         } | ||||
|         expect_info_dict(self, EXPECTED, test_video) | ||||
|  | ||||
|     def test_CSpan_playlist(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = CSpanIE(dl) | ||||
|         result = ie.extract( | ||||
|             'http://www.c-span.org/video/?318608-1/gm-ignition-switch-recall') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], '342759') | ||||
|         self.assertEqual( | ||||
|             result['title'], 'General Motors Ignition Switch Recall') | ||||
|         whole_duration = sum(e['duration'] for e in result['entries']) | ||||
|         self.assertEqual(whole_duration, 14855) | ||||
|  | ||||
|     def test_aol_playlist(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = AolIE(dl) | ||||
|         result = ie.extract( | ||||
|             'http://on.aol.com/playlist/brace-yourself---todays-weirdest-news-152147?icid=OnHomepageC4_Omg_Img#_videoid=518184316') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], '152147') | ||||
|         self.assertEqual( | ||||
|             result['title'], 'Brace Yourself - Today\'s Weirdest News') | ||||
|         self.assertTrue(len(result['entries']) >= 10) | ||||
|  | ||||
|     def test_TeacherTubeClassroom(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = TeacherTubeClassroomIE(dl) | ||||
|         result = ie.extract('http://www.teachertube.com/view_classroom.php?user=rbhagwati2') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['id'], 'rbhagwati2') | ||||
|         self.assertTrue(len(result['entries']) >= 20) | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -181,7 +181,7 @@ class TestTedSubtitles(BaseTestSubtitles): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles.keys()), 28) | ||||
|         self.assertTrue(len(subtitles.keys()) >= 28) | ||||
|  | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning(u'Automatic Captions not supported by this server') | ||||
|   | ||||
| @@ -10,6 +10,7 @@ sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| # Various small unit tests | ||||
| import io | ||||
| import json | ||||
| import xml.etree.ElementTree | ||||
|  | ||||
| #from youtube_dl.utils import htmlentity_transform | ||||
| @@ -35,6 +36,9 @@ from youtube_dl.utils import ( | ||||
|     url_basename, | ||||
|     urlencode_postdata, | ||||
|     xpath_with_ns, | ||||
|     parse_iso8601, | ||||
|     strip_jsonp, | ||||
|     uppercase_escape, | ||||
| ) | ||||
|  | ||||
| if sys.version_info < (3, 0): | ||||
| @@ -266,5 +270,19 @@ class TestUtil(unittest.TestCase): | ||||
|         data = urlencode_postdata({'username': 'foo@bar.com', 'password': '1234'}) | ||||
|         self.assertTrue(isinstance(data, bytes)) | ||||
|  | ||||
|     def test_parse_iso8601(self): | ||||
|         self.assertEqual(parse_iso8601('2014-03-23T23:04:26+0100'), 1395612266) | ||||
|         self.assertEqual(parse_iso8601('2014-03-23T22:04:26+0000'), 1395612266) | ||||
|         self.assertEqual(parse_iso8601('2014-03-23T22:04:26Z'), 1395612266) | ||||
|  | ||||
|     def test_strip_jsonp(self): | ||||
|         stripped = strip_jsonp('cb ([ {"id":"532cb",\n\n\n"x":\n3}\n]\n);') | ||||
|         d = json.loads(stripped) | ||||
|         self.assertEqual(d, [{"id": "532cb", "x": 3}]) | ||||
|  | ||||
|     def test_uppercase_escpae(self): | ||||
|         self.assertEqual(uppercase_escape(u'aä'), u'aä') | ||||
|         self.assertEqual(uppercase_escape(u'\\U0001d550'), u'𝕐') | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -112,11 +112,11 @@ class TestYoutubeLists(unittest.TestCase): | ||||
|     def test_youtube_mix(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = YoutubePlaylistIE(dl) | ||||
|         result = ie.extract('http://www.youtube.com/watch?v=lLJf9qJHR3E&list=RDrjFaenf1T-Y') | ||||
|         result = ie.extract('https://www.youtube.com/watch?v=W01L70IGBgE&index=2&list=RDOQpdSVF_k_w') | ||||
|         entries = result['entries'] | ||||
|         self.assertTrue(len(entries) >= 20) | ||||
|         original_video = entries[0] | ||||
|         self.assertEqual(original_video['id'], 'rjFaenf1T-Y') | ||||
|         self.assertEqual(original_video['id'], 'OQpdSVF_k_w') | ||||
|  | ||||
|     def test_youtube_toptracks(self): | ||||
|         print('Skipping: The playlist page gives error 500') | ||||
|   | ||||
							
								
								
									
										178
									
								
								youtube_dl/YoutubeDL.py
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										178
									
								
								youtube_dl/YoutubeDL.py
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							| @@ -8,6 +8,7 @@ import datetime | ||||
| import errno | ||||
| import io | ||||
| import json | ||||
| import locale | ||||
| import os | ||||
| import platform | ||||
| import re | ||||
| @@ -30,6 +31,7 @@ from .utils import ( | ||||
|     ContentTooShortError, | ||||
|     date_from_str, | ||||
|     DateRange, | ||||
|     DEFAULT_OUTTMPL, | ||||
|     determine_ext, | ||||
|     DownloadError, | ||||
|     encodeFilename, | ||||
| @@ -94,6 +96,7 @@ class YoutubeDL(object): | ||||
|     usenetrc:          Use netrc for authentication instead. | ||||
|     verbose:           Print additional info to stdout. | ||||
|     quiet:             Do not print messages to stdout. | ||||
|     no_warnings:       Do not print out anything for warnings. | ||||
|     forceurl:          Force printing final URL. | ||||
|     forcetitle:        Force printing title. | ||||
|     forceid:           Force printing ID. | ||||
| @@ -158,6 +161,7 @@ class YoutubeDL(object): | ||||
|     include_ads:       Download ads as well | ||||
|     default_search:    Prepend this string if an input url is not valid. | ||||
|                        'auto' for elaborate guessing | ||||
|     encoding:          Use this encoding instead of the system-specified. | ||||
|  | ||||
|     The following parameters are not used by YoutubeDL itself, they are used by | ||||
|     the FileDownloader: | ||||
| @@ -283,6 +287,9 @@ class YoutubeDL(object): | ||||
|         """Print message to stdout if not in quiet mode.""" | ||||
|         return self.to_stdout(message, skip_eol, check_quiet=True) | ||||
|  | ||||
|     def _write_string(self, s, out=None): | ||||
|         write_string(s, out=out, encoding=self.params.get('encoding')) | ||||
|  | ||||
|     def to_stdout(self, message, skip_eol=False, check_quiet=False): | ||||
|         """Print message to stdout if not in quiet mode.""" | ||||
|         if self.params.get('logger'): | ||||
| @@ -292,7 +299,7 @@ class YoutubeDL(object): | ||||
|             terminator = ['\n', ''][skip_eol] | ||||
|             output = message + terminator | ||||
|  | ||||
|             write_string(output, self._screen_file) | ||||
|             self._write_string(output, self._screen_file) | ||||
|  | ||||
|     def to_stderr(self, message): | ||||
|         """Print message to stderr.""" | ||||
| @@ -302,7 +309,7 @@ class YoutubeDL(object): | ||||
|         else: | ||||
|             message = self._bidi_workaround(message) | ||||
|             output = message + '\n' | ||||
|             write_string(output, self._err_file) | ||||
|             self._write_string(output, self._err_file) | ||||
|  | ||||
|     def to_console_title(self, message): | ||||
|         if not self.params.get('consoletitle', False): | ||||
| @@ -312,21 +319,21 @@ class YoutubeDL(object): | ||||
|             # already of type unicode() | ||||
|             ctypes.windll.kernel32.SetConsoleTitleW(ctypes.c_wchar_p(message)) | ||||
|         elif 'TERM' in os.environ: | ||||
|             write_string('\033]0;%s\007' % message, self._screen_file) | ||||
|             self._write_string('\033]0;%s\007' % message, self._screen_file) | ||||
|  | ||||
|     def save_console_title(self): | ||||
|         if not self.params.get('consoletitle', False): | ||||
|             return | ||||
|         if 'TERM' in os.environ: | ||||
|             # Save the title on stack | ||||
|             write_string('\033[22;0t', self._screen_file) | ||||
|             self._write_string('\033[22;0t', self._screen_file) | ||||
|  | ||||
|     def restore_console_title(self): | ||||
|         if not self.params.get('consoletitle', False): | ||||
|             return | ||||
|         if 'TERM' in os.environ: | ||||
|             # Restore the title from stack | ||||
|             write_string('\033[23;0t', self._screen_file) | ||||
|             self._write_string('\033[23;0t', self._screen_file) | ||||
|  | ||||
|     def __enter__(self): | ||||
|         self.save_console_title() | ||||
| @@ -376,6 +383,8 @@ class YoutubeDL(object): | ||||
|         if self.params.get('logger') is not None: | ||||
|             self.params['logger'].warning(message) | ||||
|         else: | ||||
|             if self.params.get('no_warnings'): | ||||
|                 return | ||||
|             if self._err_file.isatty() and os.name != 'nt': | ||||
|                 _msg_header = '\033[0;33mWARNING:\033[0m' | ||||
|             else: | ||||
| @@ -432,7 +441,8 @@ class YoutubeDL(object): | ||||
|                                  if v is not None) | ||||
|             template_dict = collections.defaultdict(lambda: 'NA', template_dict) | ||||
|  | ||||
|             tmpl = os.path.expanduser(self.params['outtmpl']) | ||||
|             outtmpl = self.params.get('outtmpl', DEFAULT_OUTTMPL) | ||||
|             tmpl = os.path.expanduser(outtmpl) | ||||
|             filename = tmpl % template_dict | ||||
|             return filename | ||||
|         except ValueError as err: | ||||
| @@ -697,11 +707,27 @@ class YoutubeDL(object): | ||||
|     def process_video_result(self, info_dict, download=True): | ||||
|         assert info_dict.get('_type', 'video') == 'video' | ||||
|  | ||||
|         if 'id' not in info_dict: | ||||
|             raise ExtractorError('Missing "id" field in extractor result') | ||||
|         if 'title' not in info_dict: | ||||
|             raise ExtractorError('Missing "title" field in extractor result') | ||||
|  | ||||
|         if 'playlist' not in info_dict: | ||||
|             # It isn't part of a playlist | ||||
|             info_dict['playlist'] = None | ||||
|             info_dict['playlist_index'] = None | ||||
|  | ||||
|         thumbnails = info_dict.get('thumbnails') | ||||
|         if thumbnails: | ||||
|             thumbnails.sort(key=lambda t: ( | ||||
|                 t.get('width'), t.get('height'), t.get('url'))) | ||||
|             for t in thumbnails: | ||||
|                 if 'width' in t and 'height' in t: | ||||
|                     t['resolution'] = '%dx%d' % (t['width'], t['height']) | ||||
|  | ||||
|         if thumbnails and 'thumbnail' not in info_dict: | ||||
|             info_dict['thumbnail'] = thumbnails[-1]['url'] | ||||
|  | ||||
|         if 'display_id' not in info_dict and 'id' in info_dict: | ||||
|             info_dict['display_id'] = info_dict['id'] | ||||
|  | ||||
| @@ -728,6 +754,9 @@ class YoutubeDL(object): | ||||
|  | ||||
|         # We check that all the formats have the format and format_id fields | ||||
|         for i, format in enumerate(formats): | ||||
|             if 'url' not in format: | ||||
|                 raise ExtractorError('Missing "url" key in result (index %d)' % i) | ||||
|  | ||||
|             if format.get('format_id') is None: | ||||
|                 format['format_id'] = compat_str(i) | ||||
|             if format.get('format') is None: | ||||
| @@ -738,7 +767,7 @@ class YoutubeDL(object): | ||||
|                 ) | ||||
|             # Automatically determine file extension if missing | ||||
|             if 'ext' not in format: | ||||
|                 format['ext'] = determine_ext(format['url']) | ||||
|                 format['ext'] = determine_ext(format['url']).lower() | ||||
|  | ||||
|         format_limit = self.params.get('format_limit', None) | ||||
|         if format_limit: | ||||
| @@ -863,7 +892,7 @@ class YoutubeDL(object): | ||||
|  | ||||
|         try: | ||||
|             dn = os.path.dirname(encodeFilename(filename)) | ||||
|             if dn != '' and not os.path.exists(dn): | ||||
|             if dn and not os.path.exists(dn): | ||||
|                 os.makedirs(dn) | ||||
|         except (OSError, IOError) as err: | ||||
|             self.report_error('unable to create directory ' + compat_str(err)) | ||||
| @@ -920,7 +949,7 @@ class YoutubeDL(object): | ||||
|                         with io.open(encodeFilename(sub_filename), 'w', encoding='utf-8') as subfile: | ||||
|                                 subfile.write(sub) | ||||
|                 except (OSError, IOError): | ||||
|                     self.report_error('Cannot write subtitles file ' + descfn) | ||||
|                     self.report_error('Cannot write subtitles file ' + sub_filename) | ||||
|                     return | ||||
|  | ||||
|         if self.params.get('writeinfojson', False): | ||||
| @@ -1009,10 +1038,11 @@ class YoutubeDL(object): | ||||
|  | ||||
|     def download(self, url_list): | ||||
|         """Download a given list of URLs.""" | ||||
|         outtmpl = self.params.get('outtmpl', DEFAULT_OUTTMPL) | ||||
|         if (len(url_list) > 1 and | ||||
|                 '%' not in self.params['outtmpl'] | ||||
|                 '%' not in outtmpl | ||||
|                 and self.params.get('max_downloads') != 1): | ||||
|             raise SameFileError(self.params['outtmpl']) | ||||
|             raise SameFileError(outtmpl) | ||||
|  | ||||
|         for url in url_list: | ||||
|             try: | ||||
| @@ -1123,57 +1153,57 @@ class YoutubeDL(object): | ||||
|             res = default | ||||
|         return res | ||||
|  | ||||
|     def list_formats(self, info_dict): | ||||
|         def format_note(fdict): | ||||
|             res = '' | ||||
|             if fdict.get('ext') in ['f4f', 'f4m']: | ||||
|                 res += '(unsupported) ' | ||||
|             if fdict.get('format_note') is not None: | ||||
|                 res += fdict['format_note'] + ' ' | ||||
|             if fdict.get('tbr') is not None: | ||||
|                 res += '%4dk ' % fdict['tbr'] | ||||
|             if fdict.get('container') is not None: | ||||
|                 if res: | ||||
|                     res += ', ' | ||||
|                 res += '%s container' % fdict['container'] | ||||
|             if (fdict.get('vcodec') is not None and | ||||
|                     fdict.get('vcodec') != 'none'): | ||||
|                 if res: | ||||
|                     res += ', ' | ||||
|                 res += fdict['vcodec'] | ||||
|                 if fdict.get('vbr') is not None: | ||||
|                     res += '@' | ||||
|             elif fdict.get('vbr') is not None and fdict.get('abr') is not None: | ||||
|                 res += 'video@' | ||||
|     def _format_note(self, fdict): | ||||
|         res = '' | ||||
|         if fdict.get('ext') in ['f4f', 'f4m']: | ||||
|             res += '(unsupported) ' | ||||
|         if fdict.get('format_note') is not None: | ||||
|             res += fdict['format_note'] + ' ' | ||||
|         if fdict.get('tbr') is not None: | ||||
|             res += '%4dk ' % fdict['tbr'] | ||||
|         if fdict.get('container') is not None: | ||||
|             if res: | ||||
|                 res += ', ' | ||||
|             res += '%s container' % fdict['container'] | ||||
|         if (fdict.get('vcodec') is not None and | ||||
|                 fdict.get('vcodec') != 'none'): | ||||
|             if res: | ||||
|                 res += ', ' | ||||
|             res += fdict['vcodec'] | ||||
|             if fdict.get('vbr') is not None: | ||||
|                 res += '%4dk' % fdict['vbr'] | ||||
|             if fdict.get('acodec') is not None: | ||||
|                 if res: | ||||
|                     res += ', ' | ||||
|                 if fdict['acodec'] == 'none': | ||||
|                     res += 'video only' | ||||
|                 else: | ||||
|                     res += '%-5s' % fdict['acodec'] | ||||
|             elif fdict.get('abr') is not None: | ||||
|                 if res: | ||||
|                     res += ', ' | ||||
|                 res += 'audio' | ||||
|             if fdict.get('abr') is not None: | ||||
|                 res += '@%3dk' % fdict['abr'] | ||||
|             if fdict.get('asr') is not None: | ||||
|                 res += ' (%5dHz)' % fdict['asr'] | ||||
|             if fdict.get('filesize') is not None: | ||||
|                 if res: | ||||
|                     res += ', ' | ||||
|                 res += format_bytes(fdict['filesize']) | ||||
|             return res | ||||
|                 res += '@' | ||||
|         elif fdict.get('vbr') is not None and fdict.get('abr') is not None: | ||||
|             res += 'video@' | ||||
|         if fdict.get('vbr') is not None: | ||||
|             res += '%4dk' % fdict['vbr'] | ||||
|         if fdict.get('acodec') is not None: | ||||
|             if res: | ||||
|                 res += ', ' | ||||
|             if fdict['acodec'] == 'none': | ||||
|                 res += 'video only' | ||||
|             else: | ||||
|                 res += '%-5s' % fdict['acodec'] | ||||
|         elif fdict.get('abr') is not None: | ||||
|             if res: | ||||
|                 res += ', ' | ||||
|             res += 'audio' | ||||
|         if fdict.get('abr') is not None: | ||||
|             res += '@%3dk' % fdict['abr'] | ||||
|         if fdict.get('asr') is not None: | ||||
|             res += ' (%5dHz)' % fdict['asr'] | ||||
|         if fdict.get('filesize') is not None: | ||||
|             if res: | ||||
|                 res += ', ' | ||||
|             res += format_bytes(fdict['filesize']) | ||||
|         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), | ||||
|                 format_note(format), | ||||
|                 self._format_note(format), | ||||
|             )) | ||||
|  | ||||
|         formats = info_dict.get('formats', [info_dict]) | ||||
| @@ -1181,8 +1211,8 @@ class YoutubeDL(object): | ||||
|                     max(len(f['format_id']) for f in formats)) | ||||
|         formats_s = [line(f, idlen) for f in formats] | ||||
|         if len(formats) > 1: | ||||
|             formats_s[0] += (' ' if format_note(formats[0]) else '') + '(worst)' | ||||
|             formats_s[-1] += (' ' if format_note(formats[-1]) else '') + '(best)' | ||||
|             formats_s[0] += (' ' if self._format_note(formats[0]) else '') + '(worst)' | ||||
|             formats_s[-1] += (' ' if self._format_note(formats[-1]) else '') + '(best)' | ||||
|  | ||||
|         header_line = line({ | ||||
|             'format_id': 'format code', 'ext': 'extension', | ||||
| @@ -1197,7 +1227,17 @@ class YoutubeDL(object): | ||||
|     def print_debug_header(self): | ||||
|         if not self.params.get('verbose'): | ||||
|             return | ||||
|         write_string('[debug] youtube-dl version ' + __version__ + '\n') | ||||
|  | ||||
|         write_string( | ||||
|             '[debug] Encodings: locale %s, fs %s, out %s, pref %s\n' % ( | ||||
|                 locale.getpreferredencoding(), | ||||
|                 sys.getfilesystemencoding(), | ||||
|                 sys.stdout.encoding, | ||||
|                 self.get_encoding()), | ||||
|             encoding=None | ||||
|         ) | ||||
|  | ||||
|         self._write_string('[debug] youtube-dl version ' + __version__ + '\n') | ||||
|         try: | ||||
|             sp = subprocess.Popen( | ||||
|                 ['git', 'rev-parse', '--short', 'HEAD'], | ||||
| @@ -1206,20 +1246,20 @@ class YoutubeDL(object): | ||||
|             out, err = sp.communicate() | ||||
|             out = out.decode().strip() | ||||
|             if re.match('[0-9a-f]+', out): | ||||
|                 write_string('[debug] Git HEAD: ' + out + '\n') | ||||
|                 self._write_string('[debug] Git HEAD: ' + out + '\n') | ||||
|         except: | ||||
|             try: | ||||
|                 sys.exc_clear() | ||||
|             except: | ||||
|                 pass | ||||
|         write_string('[debug] Python version %s - %s' % | ||||
|         self._write_string('[debug] Python version %s - %s' % | ||||
|                      (platform.python_version(), platform_name()) + '\n') | ||||
|  | ||||
|         proxy_map = {} | ||||
|         for handler in self._opener.handlers: | ||||
|             if hasattr(handler, 'proxies'): | ||||
|                 proxy_map.update(handler.proxies) | ||||
|         write_string('[debug] Proxy map: ' + compat_str(proxy_map) + '\n') | ||||
|         self._write_string('[debug] Proxy map: ' + compat_str(proxy_map) + '\n') | ||||
|  | ||||
|     def _setup_opener(self): | ||||
|         timeout_val = self.params.get('socket_timeout') | ||||
| @@ -1261,3 +1301,19 @@ class YoutubeDL(object): | ||||
|         # (See https://github.com/rg3/youtube-dl/issues/1309 for details) | ||||
|         opener.addheaders = [] | ||||
|         self._opener = opener | ||||
|  | ||||
|     def encode(self, s): | ||||
|         if isinstance(s, bytes): | ||||
|             return s  # Already encoded | ||||
|  | ||||
|         try: | ||||
|             return s.encode(self.get_encoding()) | ||||
|         except UnicodeEncodeError as err: | ||||
|             err.reason = err.reason + '. Check your system encoding configuration or use the --encoding option.' | ||||
|             raise | ||||
|  | ||||
|     def get_encoding(self): | ||||
|         encoding = self.params.get('encoding') | ||||
|         if encoding is None: | ||||
|             encoding = preferredencoding() | ||||
|         return encoding | ||||
|   | ||||
| @@ -51,6 +51,14 @@ __authors__  = ( | ||||
|     'David Wagner', | ||||
|     'Juan C. Olivares', | ||||
|     'Mattias Harrysson', | ||||
|     'phaer', | ||||
|     'Sainyam Kapoor', | ||||
|     'Nicolas Évrard', | ||||
|     'Jason Normore', | ||||
|     'Hoje Lee', | ||||
|     'Adam Thalhammer', | ||||
|     'Georg Jähnig', | ||||
|     'Ralf Haring', | ||||
| ) | ||||
|  | ||||
| __license__ = 'Public Domain' | ||||
| @@ -70,6 +78,7 @@ from .utils import ( | ||||
|     compat_getpass, | ||||
|     compat_print, | ||||
|     DateRange, | ||||
|     DEFAULT_OUTTMPL, | ||||
|     decodeOption, | ||||
|     get_term_width, | ||||
|     DownloadError, | ||||
| @@ -90,6 +99,8 @@ from .extractor import gen_extractors | ||||
| from .version import __version__ | ||||
| from .YoutubeDL import YoutubeDL | ||||
| from .postprocessor import ( | ||||
|     AtomicParsleyPP, | ||||
|     FFmpegAudioFixPP, | ||||
|     FFmpegMetadataPP, | ||||
|     FFmpegVideoConvertor, | ||||
|     FFmpegExtractAudioPP, | ||||
| @@ -227,6 +238,9 @@ def parseOpts(overrideArguments=None): | ||||
|     general.add_option('--referer', | ||||
|             dest='referer', help='specify a custom referer, use if the video access is restricted to one domain', | ||||
|             metavar='REF', default=None) | ||||
|     general.add_option('--add-header', | ||||
|             dest='headers', help='specify a custom HTTP header and its value, separated by a colon \':\'. You can use this option multiple times', action="append", | ||||
|             metavar='FIELD:VALUE') | ||||
|     general.add_option('--list-extractors', | ||||
|             action='store_true', dest='list_extractors', | ||||
|             help='List all supported extractors and the URLs they would handle', default=False) | ||||
| @@ -238,7 +252,7 @@ def parseOpts(overrideArguments=None): | ||||
|         help='Use the specified HTTP/HTTPS proxy. Pass in an empty string (--proxy "") for direct connection') | ||||
|     general.add_option('--no-check-certificate', action='store_true', dest='no_check_certificate', default=False, help='Suppress HTTPS certificate validation.') | ||||
|     general.add_option( | ||||
|         '--prefer-insecure', action='store_true', dest='prefer_insecure', | ||||
|         '--prefer-insecure', '--prefer-unsecure', action='store_true', dest='prefer_insecure', | ||||
|         help='Use an unencrypted connection to retrieve information about the video. (Currently supported only for YouTube)') | ||||
|     general.add_option( | ||||
|         '--cache-dir', dest='cachedir', default=get_cachedir(), metavar='DIR', | ||||
| @@ -252,13 +266,17 @@ def parseOpts(overrideArguments=None): | ||||
|     general.add_option( | ||||
|         '--bidi-workaround', dest='bidi_workaround', action='store_true', | ||||
|         help=u'Work around terminals that lack bidirectional text support. Requires bidiv or fribidi executable in PATH') | ||||
|     general.add_option('--default-search', | ||||
|             dest='default_search', metavar='PREFIX', | ||||
|             help='Use this prefix for unqualified URLs. For example "gvsearch2:" downloads two videos from google videos for  youtube-dl "large apple". By default (with value "auto") youtube-dl guesses.') | ||||
|     general.add_option( | ||||
|         '--default-search', | ||||
|         dest='default_search', metavar='PREFIX', | ||||
|         help='Use this prefix for unqualified URLs. For example "gvsearch2:" downloads two videos from google videos for  youtube-dl "large apple". By default (with value "auto") youtube-dl guesses.') | ||||
|     general.add_option( | ||||
|         '--ignore-config', | ||||
|         action='store_true', | ||||
|         help='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.conf (%APPDATA%/youtube-dl/config.txt on Windows)') | ||||
|     general.add_option( | ||||
|         '--encoding', dest='encoding', metavar='ENCODING', | ||||
|         help='Force the specified encoding (experimental)') | ||||
|  | ||||
|     selection.add_option( | ||||
|         '--playlist-start', | ||||
| @@ -361,6 +379,10 @@ def parseOpts(overrideArguments=None): | ||||
|  | ||||
|     verbosity.add_option('-q', '--quiet', | ||||
|             action='store_true', dest='quiet', help='activates quiet mode', default=False) | ||||
|     verbosity.add_option( | ||||
|         '--no-warnings', | ||||
|         dest='no_warnings', action='store_true', default=False, | ||||
|         help='Ignore warnings') | ||||
|     verbosity.add_option('-s', '--simulate', | ||||
|             action='store_true', dest='simulate', help='do not download the video and do not write anything to disk', default=False) | ||||
|     verbosity.add_option('--skip-download', | ||||
| @@ -388,7 +410,7 @@ def parseOpts(overrideArguments=None): | ||||
|             help='simulate, quiet but print output format', default=False) | ||||
|     verbosity.add_option('-j', '--dump-json', | ||||
|             action='store_true', dest='dumpjson', | ||||
|             help='simulate, quiet but print JSON information', default=False) | ||||
|             help='simulate, quiet but print JSON information. See --output for a description of available keys.', default=False) | ||||
|     verbosity.add_option('--newline', | ||||
|             action='store_true', dest='progress_with_newline', help='output progress bar as new lines', default=False) | ||||
|     verbosity.add_option('--no-progress', | ||||
| @@ -490,6 +512,8 @@ def parseOpts(overrideArguments=None): | ||||
|             help='do not overwrite post-processed files; the post-processed files are overwritten by default') | ||||
|     postproc.add_option('--embed-subs', action='store_true', dest='embedsubtitles', default=False, | ||||
|             help='embed subtitles in the video (only for mp4 videos)') | ||||
|     postproc.add_option('--embed-thumbnail', action='store_true', dest='embedthumbnail', default=False, | ||||
|             help='embed thumbnail in the audio as cover art') | ||||
|     postproc.add_option('--add-metadata', action='store_true', dest='addmetadata', default=False, | ||||
|             help='write metadata to the video file') | ||||
|     postproc.add_option('--xattrs', action='store_true', dest='xattrs', default=False, | ||||
| @@ -532,8 +556,6 @@ def parseOpts(overrideArguments=None): | ||||
|             write_string(u'[debug] System config: ' + repr(_hide_login_info(systemConf)) + '\n') | ||||
|             write_string(u'[debug] User config: ' + repr(_hide_login_info(userConf)) + '\n') | ||||
|             write_string(u'[debug] Command-line args: ' + repr(_hide_login_info(commandLineConf)) + '\n') | ||||
|             write_string(u'[debug] Encodings: locale %r, fs %r, out %r, pref: %r\n' % | ||||
|                          (locale.getpreferredencoding(), sys.getfilesystemencoding(), sys.stdout.encoding, preferredencoding())) | ||||
|  | ||||
|     return parser, opts, args | ||||
|  | ||||
| @@ -556,6 +578,16 @@ def _real_main(argv=None): | ||||
|     if opts.referer is not None: | ||||
|         std_headers['Referer'] = opts.referer | ||||
|  | ||||
|     # Custom HTTP headers | ||||
|     if opts.headers is not None: | ||||
|         for h in opts.headers: | ||||
|             if h.find(':', 1) < 0: | ||||
|                 parser.error(u'wrong header formatting, it should be key:value, not "%s"'%h) | ||||
|             key, value = h.split(':', 2) | ||||
|             if opts.verbose: | ||||
|                 write_string(u'[debug] Adding header from command line option %s:%s\n'%(key, value)) | ||||
|             std_headers[key] = value | ||||
|  | ||||
|     # Dump user agent | ||||
|     if opts.dump_user_agent: | ||||
|         compat_print(std_headers['User-Agent']) | ||||
| @@ -651,13 +683,13 @@ def _real_main(argv=None): | ||||
|         if not opts.audioquality.isdigit(): | ||||
|             parser.error(u'invalid audio quality specified') | ||||
|     if opts.recodevideo is not None: | ||||
|         if opts.recodevideo not in ['mp4', 'flv', 'webm', 'ogg']: | ||||
|         if opts.recodevideo not in ['mp4', 'flv', 'webm', 'ogg', 'mkv']: | ||||
|             parser.error(u'invalid video recode format specified') | ||||
|     if opts.date is not None: | ||||
|         date = DateRange.day(opts.date) | ||||
|     else: | ||||
|         date = DateRange(opts.dateafter, opts.datebefore) | ||||
|     if opts.default_search not in ('auto', None) and ':' not in opts.default_search: | ||||
|     if opts.default_search not in ('auto', 'auto_warning', None) and ':' not in opts.default_search: | ||||
|         parser.error(u'--default-search invalid; did you forget a colon (:) at the end?') | ||||
|  | ||||
|     # Do not download videos when there are audio-only formats | ||||
| @@ -680,7 +712,7 @@ def _real_main(argv=None): | ||||
|             or (opts.usetitle and u'%(title)s-%(id)s.%(ext)s') | ||||
|             or (opts.useid and u'%(id)s.%(ext)s') | ||||
|             or (opts.autonumber and u'%(autonumber)s-%(id)s.%(ext)s') | ||||
|             or u'%(title)s-%(id)s.%(ext)s') | ||||
|             or DEFAULT_OUTTMPL) | ||||
|     if not os.path.splitext(outtmpl)[1] and opts.extractaudio: | ||||
|         parser.error(u'Cannot download a video and extract audio into the same' | ||||
|                      u' file! Use "{0}.%(ext)s" instead of "{0}" as the output' | ||||
| @@ -695,6 +727,7 @@ def _real_main(argv=None): | ||||
|         'password': opts.password, | ||||
|         'videopassword': opts.videopassword, | ||||
|         'quiet': (opts.quiet or any_printing), | ||||
|         'no_warnings': opts.no_warnings, | ||||
|         'forceurl': opts.geturl, | ||||
|         'forcetitle': opts.gettitle, | ||||
|         'forceid': opts.getid, | ||||
| @@ -767,6 +800,7 @@ def _real_main(argv=None): | ||||
|         'include_ads': opts.include_ads, | ||||
|         'default_search': opts.default_search, | ||||
|         'youtube_include_dash_manifest': opts.youtube_include_dash_manifest, | ||||
|         'encoding': opts.encoding, | ||||
|     } | ||||
|  | ||||
|     with YoutubeDL(ydl_opts) as ydl: | ||||
| @@ -785,6 +819,10 @@ def _real_main(argv=None): | ||||
|             ydl.add_post_processor(FFmpegEmbedSubtitlePP(subtitlesformat=opts.subtitlesformat)) | ||||
|         if opts.xattrs: | ||||
|             ydl.add_post_processor(XAttrMetadataPP()) | ||||
|         if opts.embedthumbnail: | ||||
|             if not opts.addmetadata: | ||||
|                 ydl.add_post_processor(FFmpegAudioFixPP()) | ||||
|             ydl.add_post_processor(AtomicParsleyPP()) | ||||
|  | ||||
|         # Update version | ||||
|         if opts.update_self: | ||||
|   | ||||
| @@ -4,9 +4,10 @@ import sys | ||||
| import time | ||||
|  | ||||
| from ..utils import ( | ||||
|     compat_str, | ||||
|     encodeFilename, | ||||
|     timeconvert, | ||||
|     format_bytes, | ||||
|     timeconvert, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -173,7 +174,7 @@ class FileDownloader(object): | ||||
|                 return | ||||
|             os.rename(encodeFilename(old_filename), encodeFilename(new_filename)) | ||||
|         except (IOError, OSError) as err: | ||||
|             self.report_error(u'unable to rename file: %s' % str(err)) | ||||
|             self.report_error(u'unable to rename file: %s' % compat_str(err)) | ||||
|  | ||||
|     def try_utime(self, filename, last_modified_hdr): | ||||
|         """Try to set the last-modified time of the given file.""" | ||||
|   | ||||
| @@ -297,6 +297,7 @@ class F4mFD(FileDownloader): | ||||
|                         break | ||||
|             frags_filenames.append(frag_filename) | ||||
|  | ||||
|         dest_stream.close() | ||||
|         self.report_finish(format_bytes(state['downloaded_bytes']), time.time() - start) | ||||
|  | ||||
|         self.try_rename(tmpfilename, filename) | ||||
|   | ||||
| @@ -13,8 +13,10 @@ class HlsFD(FileDownloader): | ||||
|         self.report_destination(filename) | ||||
|         tmpfilename = self.temp_name(filename) | ||||
|  | ||||
|         args = ['-y', '-i', url, '-f', 'mp4', '-c', 'copy', | ||||
|             '-bsf:a', 'aac_adtstoasc', tmpfilename] | ||||
|         args = [ | ||||
|             '-y', '-i', url, '-f', 'mp4', '-c', 'copy', | ||||
|             '-bsf:a', 'aac_adtstoasc', | ||||
|             encodeFilename(tmpfilename, for_subprocess=True)] | ||||
|  | ||||
|         for program in ['avconv', 'ffmpeg']: | ||||
|             try: | ||||
| @@ -23,7 +25,7 @@ class HlsFD(FileDownloader): | ||||
|             except (OSError, IOError): | ||||
|                 pass | ||||
|         else: | ||||
|             self.report_error(u'm3u8 download detected but ffmpeg or avconv could not be found') | ||||
|             self.report_error(u'm3u8 download detected but ffmpeg or avconv could not be found. Please install one.') | ||||
|         cmd = [program] + args | ||||
|  | ||||
|         retval = subprocess.call(cmd) | ||||
|   | ||||
| @@ -14,6 +14,8 @@ from ..utils import ( | ||||
|  | ||||
|  | ||||
| class HttpFD(FileDownloader): | ||||
|     _TEST_FILE_SIZE = 10241 | ||||
|  | ||||
|     def real_download(self, filename, info_dict): | ||||
|         url = info_dict['url'] | ||||
|         tmpfilename = self.temp_name(filename) | ||||
| @@ -23,11 +25,15 @@ class HttpFD(FileDownloader): | ||||
|         headers = {'Youtubedl-no-compression': 'True'} | ||||
|         if 'user_agent' in info_dict: | ||||
|             headers['Youtubedl-user-agent'] = info_dict['user_agent'] | ||||
|         if 'http_referer' in info_dict: | ||||
|             headers['Referer'] = info_dict['http_referer'] | ||||
|         basic_request = compat_urllib_request.Request(url, None, headers) | ||||
|         request = compat_urllib_request.Request(url, None, headers) | ||||
|  | ||||
|         if self.params.get('test', False): | ||||
|             request.add_header('Range', 'bytes=0-10240') | ||||
|         is_test = self.params.get('test', False) | ||||
|  | ||||
|         if is_test: | ||||
|             request.add_header('Range', 'bytes=0-%s' % str(self._TEST_FILE_SIZE - 1)) | ||||
|  | ||||
|         # Establish possible resume length | ||||
|         if os.path.isfile(encodeFilename(tmpfilename)): | ||||
| @@ -98,6 +104,15 @@ class HttpFD(FileDownloader): | ||||
|             return False | ||||
|  | ||||
|         data_len = data.info().get('Content-length', None) | ||||
|  | ||||
|         # Range HTTP header may be ignored/unsupported by a webserver | ||||
|         # (e.g. extractor/scivee.py, extractor/bambuser.py). | ||||
|         # However, for a test we still would like to download just a piece of a file. | ||||
|         # To achieve this we limit data_len to _TEST_FILE_SIZE and manually control | ||||
|         # block size when downloading a file. | ||||
|         if is_test and (data_len is None or int(data_len) > self._TEST_FILE_SIZE): | ||||
|             data_len = self._TEST_FILE_SIZE | ||||
|  | ||||
|         if data_len is not None: | ||||
|             data_len = int(data_len) + resume_len | ||||
|             min_data_len = self.params.get("min_filesize", None) | ||||
| @@ -116,7 +131,7 @@ class HttpFD(FileDownloader): | ||||
|         while True: | ||||
|             # Download and write | ||||
|             before = time.time() | ||||
|             data_block = data.read(block_size) | ||||
|             data_block = data.read(block_size if not is_test else min(block_size, data_len - byte_counter)) | ||||
|             after = time.time() | ||||
|             if len(data_block) == 0: | ||||
|                 break | ||||
| @@ -160,6 +175,9 @@ class HttpFD(FileDownloader): | ||||
|                 'speed': speed, | ||||
|             }) | ||||
|  | ||||
|             if is_test and byte_counter == data_len: | ||||
|                 break | ||||
|  | ||||
|             # Apply rate limit | ||||
|             self.slow_down(start, byte_counter - resume_len) | ||||
|  | ||||
|   | ||||
| @@ -10,6 +10,7 @@ from .common import FileDownloader | ||||
| from ..utils import ( | ||||
|     encodeFilename, | ||||
|     format_bytes, | ||||
|     compat_str, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -95,6 +96,7 @@ class RtmpFD(FileDownloader): | ||||
|         flash_version = info_dict.get('flash_version', None) | ||||
|         live = info_dict.get('rtmp_live', False) | ||||
|         conn = info_dict.get('rtmp_conn', None) | ||||
|         protocol = info_dict.get('rtmp_protocol', None) | ||||
|  | ||||
|         self.report_destination(filename) | ||||
|         tmpfilename = self.temp_name(filename) | ||||
| @@ -104,7 +106,7 @@ class RtmpFD(FileDownloader): | ||||
|         try: | ||||
|             subprocess.call(['rtmpdump', '-h'], stdout=(open(os.path.devnull, 'w')), stderr=subprocess.STDOUT) | ||||
|         except (OSError, IOError): | ||||
|             self.report_error('RTMP download detected but "rtmpdump" could not be run') | ||||
|             self.report_error('RTMP download detected but "rtmpdump" could not be run. Please install it.') | ||||
|             return False | ||||
|  | ||||
|         # Download using rtmpdump. rtmpdump returns exit code 2 when | ||||
| @@ -127,8 +129,13 @@ class RtmpFD(FileDownloader): | ||||
|             basic_args += ['--flashVer', flash_version] | ||||
|         if live: | ||||
|             basic_args += ['--live'] | ||||
|         if conn: | ||||
|         if isinstance(conn, list): | ||||
|             for entry in conn: | ||||
|                 basic_args += ['--conn', entry] | ||||
|         elif isinstance(conn, compat_str): | ||||
|             basic_args += ['--conn', conn] | ||||
|         if protocol is not None: | ||||
|             basic_args += ['--protocol', protocol] | ||||
|         args = basic_args + [[], ['--resume', '--skip', '1']][not live and self.params.get('continuedl', False)] | ||||
|  | ||||
|         if sys.platform == 'win32' and sys.version_info < (3, 0): | ||||
|   | ||||
| @@ -14,22 +14,26 @@ from .arte import ( | ||||
|     ArteTVConcertIE, | ||||
|     ArteTVFutureIE, | ||||
|     ArteTVDDCIE, | ||||
|     ArteTVEmbedIE, | ||||
| ) | ||||
| from .auengine import AUEngineIE | ||||
| from .bambuser import BambuserIE, BambuserChannelIE | ||||
| from .bandcamp import BandcampIE, BandcampAlbumIE | ||||
| from .bbccouk import BBCCoUkIE | ||||
| from .bilibili import BiliBiliIE | ||||
| from .blinkx import BlinkxIE | ||||
| from .bliptv import BlipTVIE, BlipTVUserIE | ||||
| from .bloomberg import BloombergIE | ||||
| from .br import BRIE | ||||
| from .breakcom import BreakIE | ||||
| from .brightcove import BrightcoveIE | ||||
| from .byutv import BYUtvIE | ||||
| from .c56 import C56IE | ||||
| from .canal13cl import Canal13clIE | ||||
| from .canalplus import CanalplusIE | ||||
| from .canalc2 import Canalc2IE | ||||
| from .cbs import CBSIE | ||||
| from .cbsnews import CBSNewsIE | ||||
| from .ceskatelevize import CeskaTelevizeIE | ||||
| from .channel9 import Channel9IE | ||||
| from .chilloutzone import ChilloutzoneIE | ||||
| @@ -37,7 +41,9 @@ from .cinemassacre import CinemassacreIE | ||||
| from .clipfish import ClipfishIE | ||||
| from .cliphunter import CliphunterIE | ||||
| from .clipsyndicate import ClipsyndicateIE | ||||
| from .clubic import ClubicIE | ||||
| from .cmt import CMTIE | ||||
| from .cnet import CNETIE | ||||
| from .cnn import ( | ||||
|     CNNIE, | ||||
|     CNNBlogsIE, | ||||
| @@ -59,12 +65,14 @@ from .dotsub import DotsubIE | ||||
| from .dreisat import DreiSatIE | ||||
| from .defense import DefenseGouvFrIE | ||||
| from .discovery import DiscoveryIE | ||||
| from .divxstage import DivxStageIE | ||||
| from .dropbox import DropboxIE | ||||
| from .ebaumsworld import EbaumsWorldIE | ||||
| from .ehow import EHowIE | ||||
| from .eighttracks import EightTracksIE | ||||
| from .eitb import EitbIE | ||||
| from .elpais import ElPaisIE | ||||
| from .empflix import EmpflixIE | ||||
| from .engadget import EngadgetIE | ||||
| from .escapist import EscapistIE | ||||
| from .everyonesmixtape import EveryonesMixtapeIE | ||||
| @@ -72,6 +80,7 @@ from .exfm import ExfmIE | ||||
| from .extremetube import ExtremeTubeIE | ||||
| from .facebook import FacebookIE | ||||
| from .faz import FazIE | ||||
| from .fc2 import FC2IE | ||||
| from .firstpost import FirstpostIE | ||||
| from .firsttv import FirstTVIE | ||||
| from .fivemin import FiveMinIE | ||||
| @@ -81,6 +90,7 @@ from .fktv import ( | ||||
| ) | ||||
| from .flickr import FlickrIE | ||||
| from .fourtube import FourTubeIE | ||||
| from .franceculture import FranceCultureIE | ||||
| from .franceinter import FranceInterIE | ||||
| from .francetv import ( | ||||
|     PluzzIE, | ||||
| @@ -99,12 +109,15 @@ from .gdcvault import GDCVaultIE | ||||
| from .generic import GenericIE | ||||
| from .googleplus import GooglePlusIE | ||||
| from .googlesearch import GoogleSearchIE | ||||
| from .gorillavid import GorillaVidIE | ||||
| from .hark import HarkIE | ||||
| from .helsinki import HelsinkiIE | ||||
| from .hentaistigma import HentaiStigmaIE | ||||
| from .hotnewhiphop import HotNewHipHopIE | ||||
| from .howcast import HowcastIE | ||||
| from .huffpost import HuffPostIE | ||||
| from .hypem import HypemIE | ||||
| from .iconosquare import IconosquareIE | ||||
| from .ign import IGNIE, OneUPIE | ||||
| from .imdb import ( | ||||
|     ImdbIE, | ||||
| @@ -130,10 +143,15 @@ from .khanacademy import KhanAcademyIE | ||||
| from .kickstarter import KickStarterIE | ||||
| from .keek import KeekIE | ||||
| from .kontrtube import KontrTubeIE | ||||
| from .ku6 import Ku6IE | ||||
| from .la7 import LA7IE | ||||
| from .lifenews import LifeNewsIE | ||||
| from .liveleak import LiveLeakIE | ||||
| from .livestream import LivestreamIE, LivestreamOriginalIE | ||||
| from .livestream import ( | ||||
|     LivestreamIE, | ||||
|     LivestreamOriginalIE, | ||||
|     LivestreamShortenerIE, | ||||
| ) | ||||
| from .lynda import ( | ||||
|     LyndaIE, | ||||
|     LyndaCourseIE | ||||
| @@ -150,10 +168,16 @@ from .mixcloud import MixcloudIE | ||||
| from .mpora import MporaIE | ||||
| from .mofosex import MofosexIE | ||||
| from .mooshare import MooshareIE | ||||
| from .morningstar import MorningstarIE | ||||
| from .motorsport import MotorsportIE | ||||
| from .moviezine import MoviezineIE | ||||
| from .movshare import MovShareIE | ||||
| from .mtv import ( | ||||
|     MTVIE, | ||||
|     MTVServicesEmbeddedIE, | ||||
|     MTVIggyIE, | ||||
| ) | ||||
| from .musicplayon import MusicPlayOnIE | ||||
| from .muzu import MuzuTVIE | ||||
| from .myspace import MySpaceIE | ||||
| from .myspass import MySpassIE | ||||
| @@ -167,14 +191,24 @@ from .nbc import ( | ||||
| from .ndr import NDRIE | ||||
| from .ndtv import NDTVIE | ||||
| from .newgrounds import NewgroundsIE | ||||
| from .newstube import NewstubeIE | ||||
| from .nfb import NFBIE | ||||
| from .nhl import NHLIE, NHLVideocenterIE | ||||
| from .niconico import NiconicoIE | ||||
| from .ninegag import NineGagIE | ||||
| from .noco import NocoIE | ||||
| from .normalboots import NormalbootsIE | ||||
| from .novamov import NovaMovIE | ||||
| from .nowness import NownessIE | ||||
| from .nowvideo import NowVideoIE | ||||
| from .nrk import ( | ||||
|     NRKIE, | ||||
|     NRKTVIE, | ||||
| ) | ||||
| from .ntv import NTVIE | ||||
| from .nytimes import NYTimesIE | ||||
| from .nuvid import NuvidIE | ||||
| from .oe1 import OE1IE | ||||
| from .ooyala import OoyalaIE | ||||
| from .orf import ORFIE | ||||
| from .parliamentliveuk import ParliamentLiveUKIE | ||||
| @@ -188,13 +222,17 @@ from .pornotube import PornotubeIE | ||||
| from .prosiebensat1 import ProSiebenSat1IE | ||||
| from .pyvideo import PyvideoIE | ||||
| from .radiofrance import RadioFranceIE | ||||
| from .rai import RaiIE | ||||
| from .rbmaradio import RBMARadioIE | ||||
| from .redtube import RedTubeIE | ||||
| from .ringtv import RingTVIE | ||||
| from .ro220 import Ro220IE | ||||
| from .rottentomatoes import RottenTomatoesIE | ||||
| from .roxwel import RoxwelIE | ||||
| from .rtbf import RTBFIE | ||||
| from .rtlnow import RTLnowIE | ||||
| from .rts import RTSIE | ||||
| from .rtve import RTVEALaCartaIE | ||||
| from .rutube import ( | ||||
|     RutubeIE, | ||||
|     RutubeChannelIE, | ||||
| @@ -203,10 +241,11 @@ from .rutube import ( | ||||
| ) | ||||
| from .rutv import RUTVIE | ||||
| from .savefrom import SaveFromIE | ||||
| from .scivee import SciVeeIE | ||||
| from .servingsys import ServingSysIE | ||||
| from .sina import SinaIE | ||||
| from .slashdot import SlashdotIE | ||||
| from .slideshare import SlideshareIE | ||||
| from .slutload import SlutloadIE | ||||
| from .smotri import ( | ||||
|     SmotriIE, | ||||
|     SmotriCommunityIE, | ||||
| @@ -214,7 +253,13 @@ from .smotri import ( | ||||
|     SmotriBroadcastIE, | ||||
| ) | ||||
| from .sohu import SohuIE | ||||
| from .soundcloud import SoundcloudIE, SoundcloudSetIE, SoundcloudUserIE | ||||
| from .soundcloud import ( | ||||
|     SoundcloudIE, | ||||
|     SoundcloudSetIE, | ||||
|     SoundcloudUserIE, | ||||
|     SoundcloudPlaylistIE | ||||
| ) | ||||
| from .soundgasm import SoundgasmIE | ||||
| from .southparkstudios import ( | ||||
|     SouthParkStudiosIE, | ||||
|     SouthparkDeIE, | ||||
| @@ -222,14 +267,21 @@ from .southparkstudios import ( | ||||
| from .space import SpaceIE | ||||
| from .spankwire import SpankwireIE | ||||
| from .spiegel import SpiegelIE | ||||
| from .spiegeltv import SpiegeltvIE | ||||
| from .spike import SpikeIE | ||||
| from .stanfordoc import StanfordOpenClassroomIE | ||||
| from .statigram import StatigramIE | ||||
| from .steam import SteamIE | ||||
| from .streamcloud import StreamcloudIE | ||||
| from .streamcz import StreamCZIE | ||||
| from .swrmediathek import SWRMediathekIE | ||||
| from .syfy import SyfyIE | ||||
| from .sztvhu import SztvHuIE | ||||
| from .tagesschau import TagesschauIE | ||||
| from .teachertube import ( | ||||
|     TeacherTubeIE, | ||||
|     TeacherTubeClassroomIE, | ||||
| ) | ||||
| from .teachingchannel import TeachingChannelIE | ||||
| from .teamcoco import TeamcocoIE | ||||
| from .techtalks import TechTalksIE | ||||
| from .ted import TEDIE | ||||
| @@ -238,6 +290,7 @@ from .tf1 import TF1IE | ||||
| from .theplatform import ThePlatformIE | ||||
| from .thisav import ThisAVIE | ||||
| from .tinypic import TinyPicIE | ||||
| from .tlc import TlcIE, TlcDeIE | ||||
| from .toutv import TouTvIE | ||||
| from .toypics import ToypicsUserIE, ToypicsIE | ||||
| from .traileraddict import TrailerAddictIE | ||||
| @@ -254,19 +307,22 @@ from .udemy import ( | ||||
|     UdemyCourseIE | ||||
| ) | ||||
| from .unistra import UnistraIE | ||||
| from .urort import UrortIE | ||||
| from .ustream import UstreamIE, UstreamChannelIE | ||||
| from .vbox7 import Vbox7IE | ||||
| from .veehd import VeeHDIE | ||||
| from .veoh import VeohIE | ||||
| from .vesti import VestiIE | ||||
| from .vevo import VevoIE | ||||
| from .vice import ViceIE | ||||
| from .vh1 import VH1IE | ||||
| from .viddler import ViddlerIE | ||||
| from .videobam import VideoBamIE | ||||
| from .videodetective import VideoDetectiveIE | ||||
| from .videolecturesnet import VideoLecturesNetIE | ||||
| from .videofyme import VideofyMeIE | ||||
| from .videopremium import VideoPremiumIE | ||||
| from .videott import VideoTtIE | ||||
| from .videoweed import VideoWeedIE | ||||
| from .vimeo import ( | ||||
|     VimeoIE, | ||||
|     VimeoChannelIE, | ||||
| @@ -274,17 +330,29 @@ from .vimeo import ( | ||||
|     VimeoAlbumIE, | ||||
|     VimeoGroupsIE, | ||||
|     VimeoReviewIE, | ||||
|     VimeoWatchLaterIE, | ||||
| ) | ||||
| from .vine import ( | ||||
|     VineIE, | ||||
|     VineUserIE, | ||||
| ) | ||||
| from .vine import VineIE | ||||
| from .viki import VikiIE | ||||
| from .vk import VKIE | ||||
| from .vube import VubeIE | ||||
| from .vuclip import VuClipIE | ||||
| from .vulture import VultureIE | ||||
| from .washingtonpost import WashingtonPostIE | ||||
| from .wat import WatIE | ||||
| from .wdr import WDRIE | ||||
| from .wdr import ( | ||||
|     WDRIE, | ||||
|     WDRMobileIE, | ||||
|     WDRMausIE, | ||||
| ) | ||||
| from .weibo import WeiboIE | ||||
| from .wimp import WimpIE | ||||
| from .wistia import WistiaIE | ||||
| from .worldstarhiphop import WorldStarHipHopIE | ||||
| from .wrzuta import WrzutaIE | ||||
| from .xbef import XBefIE | ||||
| from .xhamster import XHamsterIE | ||||
| from .xnxx import XNXXIE | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| @@ -14,14 +16,14 @@ from ..utils import ( | ||||
| class AddAnimeIE(InfoExtractor): | ||||
|  | ||||
|     _VALID_URL = r'^http://(?:\w+\.)?add-anime\.net/watch_video\.php\?(?:.*?)v=(?P<video_id>[\w_]+)(?:.*)' | ||||
|     IE_NAME = u'AddAnime' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.add-anime.net/watch_video.php?v=24MR3YO5SAS9', | ||||
|         u'file': u'24MR3YO5SAS9.mp4', | ||||
|         u'md5': u'72954ea10bc979ab5e2eb288b21425a0', | ||||
|         u'info_dict': { | ||||
|             u"description": u"One Piece 606", | ||||
|             u"title": u"One Piece 606" | ||||
|         'url': 'http://www.add-anime.net/watch_video.php?v=24MR3YO5SAS9', | ||||
|         'md5': '72954ea10bc979ab5e2eb288b21425a0', | ||||
|         'info_dict': { | ||||
|             'id': '24MR3YO5SAS9', | ||||
|             'ext': 'mp4', | ||||
|             'description': 'One Piece 606', | ||||
|             'title': 'One Piece 606', | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -38,10 +40,10 @@ class AddAnimeIE(InfoExtractor): | ||||
|             redir_webpage = ee.cause.read().decode('utf-8') | ||||
|             action = self._search_regex( | ||||
|                 r'<form id="challenge-form" action="([^"]+)"', | ||||
|                 redir_webpage, u'Redirect form') | ||||
|                 redir_webpage, 'Redirect form') | ||||
|             vc = self._search_regex( | ||||
|                 r'<input type="hidden" name="jschl_vc" value="([^"]+)"/>', | ||||
|                 redir_webpage, u'redirect vc value') | ||||
|                 redir_webpage, 'redirect vc value') | ||||
|             av = re.search( | ||||
|                 r'a\.value = ([0-9]+)[+]([0-9]+)[*]([0-9]+);', | ||||
|                 redir_webpage) | ||||
| @@ -52,19 +54,19 @@ class AddAnimeIE(InfoExtractor): | ||||
|             parsed_url = compat_urllib_parse_urlparse(url) | ||||
|             av_val = av_res + len(parsed_url.netloc) | ||||
|             confirm_url = ( | ||||
|                 parsed_url.scheme + u'://' + parsed_url.netloc + | ||||
|                 parsed_url.scheme + '://' + parsed_url.netloc + | ||||
|                 action + '?' + | ||||
|                 compat_urllib_parse.urlencode({ | ||||
|                     'jschl_vc': vc, 'jschl_answer': compat_str(av_val)})) | ||||
|             self._download_webpage( | ||||
|                 confirm_url, video_id, | ||||
|                 note=u'Confirming after redirect') | ||||
|                 note='Confirming after redirect') | ||||
|             webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         formats = [] | ||||
|         for format_id in ('normal', 'hq'): | ||||
|             rex = r"var %s_video_file = '(.*?)';" % re.escape(format_id) | ||||
|             video_url = self._search_regex(rex, webpage, u'video file URLx', | ||||
|             video_url = self._search_regex(rex, webpage, 'video file URLx', | ||||
|                                            fatal=False) | ||||
|             if not video_url: | ||||
|                 continue | ||||
| @@ -72,14 +74,13 @@ class AddAnimeIE(InfoExtractor): | ||||
|                 'format_id': format_id, | ||||
|                 'url': video_url, | ||||
|             }) | ||||
|         if not formats: | ||||
|             raise ExtractorError(u'Cannot find any video format!') | ||||
|         self._sort_formats(formats) | ||||
|         video_title = self._og_search_title(webpage) | ||||
|         video_description = self._og_search_description(webpage) | ||||
|  | ||||
|         return { | ||||
|             '_type': 'video', | ||||
|             'id':  video_id, | ||||
|             'id': video_id, | ||||
|             'formats': formats, | ||||
|             'title': video_title, | ||||
|             'description': video_description | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| # encoding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import datetime | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| @@ -16,6 +15,7 @@ class AftonbladetIE(InfoExtractor): | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Vulkanutbrott i rymden - nu släpper NASA bilderna', | ||||
|             'description': 'Jupiters måne mest aktiv av alla himlakroppar', | ||||
|             'timestamp': 1394142732, | ||||
|             'upload_date': '20140306', | ||||
|         }, | ||||
|     } | ||||
| @@ -27,17 +27,17 @@ class AftonbladetIE(InfoExtractor): | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         # find internal video meta data | ||||
|         META_URL = 'http://aftonbladet-play.drlib.aptoma.no/video/%s.json' | ||||
|         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') | ||||
|         internal_meta_url = META_URL % internal_meta_id | ||||
|         internal_meta_url = meta_url % internal_meta_id | ||||
|         internal_meta_json = self._download_json( | ||||
|             internal_meta_url, video_id, 'Downloading video meta data') | ||||
|  | ||||
|         # find internal video formats | ||||
|         FORMATS_URL = 'http://aftonbladet-play.videodata.drvideo.aptoma.no/actions/video/?id=%s' | ||||
|         format_url = 'http://aftonbladet-play.videodata.drvideo.aptoma.no/actions/video/?id=%s' | ||||
|         internal_video_id = internal_meta_json['videoId'] | ||||
|         internal_formats_url = FORMATS_URL % internal_video_id | ||||
|         internal_formats_url = format_url % internal_video_id | ||||
|         internal_formats_json = self._download_json( | ||||
|             internal_formats_url, video_id, 'Downloading video formats') | ||||
|  | ||||
| @@ -54,16 +54,13 @@ class AftonbladetIE(InfoExtractor): | ||||
|             }) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         timestamp = datetime.datetime.fromtimestamp(internal_meta_json['timePublished']) | ||||
|         upload_date = timestamp.strftime('%Y%m%d') | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': internal_meta_json['title'], | ||||
|             'formats': formats, | ||||
|             'thumbnail': internal_meta_json['imageUrl'], | ||||
|             'description': internal_meta_json['shortPreamble'], | ||||
|             'upload_date': upload_date, | ||||
|             'timestamp': internal_meta_json['timePublished'], | ||||
|             'duration': internal_meta_json['duration'], | ||||
|             'view_count': internal_meta_json['views'], | ||||
|         } | ||||
|   | ||||
| @@ -8,7 +8,18 @@ from .fivemin import FiveMinIE | ||||
|  | ||||
| class AolIE(InfoExtractor): | ||||
|     IE_NAME = 'on.aol.com' | ||||
|     _VALID_URL = r'http://on\.aol\.com/video/.*-(?P<id>\d+)($|\?)' | ||||
|     _VALID_URL = r'''(?x) | ||||
|         (?: | ||||
|             aol-video:| | ||||
|             http://on\.aol\.com/ | ||||
|             (?: | ||||
|                 video/.*-| | ||||
|                 playlist/(?P<playlist_display_id>[^/?#]+?)-(?P<playlist_id>[0-9]+)[?#].*_videoid= | ||||
|             ) | ||||
|         ) | ||||
|         (?P<id>[0-9]+) | ||||
|         (?:$|\?) | ||||
|     ''' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://on.aol.com/video/u-s--official-warns-of-largest-ever-irs-phone-scam-518167793?icid=OnHomepageC2Wide_MustSee_Img', | ||||
| @@ -24,5 +35,31 @@ class AolIE(InfoExtractor): | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         self.to_screen('Downloading 5min.com video %s' % video_id) | ||||
|  | ||||
|         playlist_id = mobj.group('playlist_id') | ||||
|         if playlist_id and not self._downloader.params.get('noplaylist'): | ||||
|             self.to_screen('Downloading playlist %s - add --no-playlist to just download video %s' % (playlist_id, video_id)) | ||||
|  | ||||
|             webpage = self._download_webpage(url, playlist_id) | ||||
|             title = self._html_search_regex( | ||||
|                 r'<h1 class="video-title[^"]*">(.+?)</h1>', webpage, 'title') | ||||
|             playlist_html = self._search_regex( | ||||
|                 r"(?s)<ul\s+class='video-related[^']*'>(.*?)</ul>", webpage, | ||||
|                 'playlist HTML') | ||||
|             entries = [{ | ||||
|                 '_type': 'url', | ||||
|                 'url': 'aol-video:%s' % m.group('id'), | ||||
|                 'ie_key': 'Aol', | ||||
|             } for m in re.finditer( | ||||
|                 r"<a\s+href='.*videoid=(?P<id>[0-9]+)'\s+class='video-thumb'>", | ||||
|                 playlist_html)] | ||||
|  | ||||
|             return { | ||||
|                 '_type': 'playlist', | ||||
|                 'id': playlist_id, | ||||
|                 'display_id': mobj.group('playlist_display_id'), | ||||
|                 'title': title, | ||||
|                 'entries': entries, | ||||
|             } | ||||
|  | ||||
|         return FiveMinIE._build_result(video_id) | ||||
|   | ||||
| @@ -6,7 +6,6 @@ import json | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urlparse, | ||||
|     determine_ext, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -16,9 +15,10 @@ class AppleTrailersIE(InfoExtractor): | ||||
|         "url": "http://trailers.apple.com/trailers/wb/manofsteel/", | ||||
|         "playlist": [ | ||||
|             { | ||||
|                 "file": "manofsteel-trailer4.mov", | ||||
|                 "md5": "d97a8e575432dbcb81b7c3acb741f8a8", | ||||
|                 "info_dict": { | ||||
|                     "id": "manofsteel-trailer4", | ||||
|                     "ext": "mov", | ||||
|                     "duration": 111, | ||||
|                     "title": "Trailer 4", | ||||
|                     "upload_date": "20130523", | ||||
| @@ -26,9 +26,10 @@ class AppleTrailersIE(InfoExtractor): | ||||
|                 }, | ||||
|             }, | ||||
|             { | ||||
|                 "file": "manofsteel-trailer3.mov", | ||||
|                 "md5": "b8017b7131b721fb4e8d6f49e1df908c", | ||||
|                 "info_dict": { | ||||
|                     "id": "manofsteel-trailer3", | ||||
|                     "ext": "mov", | ||||
|                     "duration": 182, | ||||
|                     "title": "Trailer 3", | ||||
|                     "upload_date": "20130417", | ||||
| @@ -36,9 +37,10 @@ class AppleTrailersIE(InfoExtractor): | ||||
|                 }, | ||||
|             }, | ||||
|             { | ||||
|                 "file": "manofsteel-trailer.mov", | ||||
|                 "md5": "d0f1e1150989b9924679b441f3404d48", | ||||
|                 "info_dict": { | ||||
|                     "id": "manofsteel-trailer", | ||||
|                     "ext": "mov", | ||||
|                     "duration": 148, | ||||
|                     "title": "Trailer", | ||||
|                     "upload_date": "20121212", | ||||
| @@ -46,15 +48,16 @@ class AppleTrailersIE(InfoExtractor): | ||||
|                 }, | ||||
|             }, | ||||
|             { | ||||
|                 "file": "manofsteel-teaser.mov", | ||||
|                 "md5": "5fe08795b943eb2e757fa95cb6def1cb", | ||||
|                 "info_dict": { | ||||
|                     "id": "manofsteel-teaser", | ||||
|                     "ext": "mov", | ||||
|                     "duration": 93, | ||||
|                     "title": "Teaser", | ||||
|                     "upload_date": "20120721", | ||||
|                     "uploader_id": "wb", | ||||
|                 }, | ||||
|             } | ||||
|             }, | ||||
|         ] | ||||
|     } | ||||
|  | ||||
| @@ -65,16 +68,16 @@ class AppleTrailersIE(InfoExtractor): | ||||
|         movie = mobj.group('movie') | ||||
|         uploader_id = mobj.group('company') | ||||
|  | ||||
|         playlist_url = compat_urlparse.urljoin(url, u'includes/playlists/itunes.inc') | ||||
|         playlist_url = compat_urlparse.urljoin(url, 'includes/playlists/itunes.inc') | ||||
|         def fix_html(s): | ||||
|             s = re.sub(r'(?s)<script[^<]*?>.*?</script>', u'', s) | ||||
|             s = re.sub(r'(?s)<script[^<]*?>.*?</script>', '', s) | ||||
|             s = re.sub(r'<img ([^<]*?)>', r'<img \1/>', s) | ||||
|             # The ' in the onClick attributes are not escaped, it couldn't be parsed | ||||
|             # like: http://trailers.apple.com/trailers/wb/gravity/ | ||||
|             def _clean_json(m): | ||||
|                 return u'iTunes.playURL(%s);' % m.group(1).replace('\'', ''') | ||||
|                 return 'iTunes.playURL(%s);' % m.group(1).replace('\'', ''') | ||||
|             s = re.sub(self._JSON_RE, _clean_json, s) | ||||
|             s = u'<html>' + s + u'</html>' | ||||
|             s = '<html>' + s + u'</html>' | ||||
|             return s | ||||
|         doc = self._download_xml(playlist_url, movie, transform_source=fix_html) | ||||
|  | ||||
| @@ -82,7 +85,7 @@ class AppleTrailersIE(InfoExtractor): | ||||
|         for li in doc.findall('./div/ul/li'): | ||||
|             on_click = li.find('.//a').attrib['onClick'] | ||||
|             trailer_info_json = self._search_regex(self._JSON_RE, | ||||
|                 on_click, u'trailer info') | ||||
|                 on_click, 'trailer info') | ||||
|             trailer_info = json.loads(trailer_info_json) | ||||
|             title = trailer_info['title'] | ||||
|             video_id = movie + '-' + re.sub(r'[^a-zA-Z0-9]', '', title).lower() | ||||
| @@ -98,8 +101,7 @@ class AppleTrailersIE(InfoExtractor): | ||||
|             first_url = trailer_info['url'] | ||||
|             trailer_id = first_url.split('/')[-1].rpartition('_')[0].lower() | ||||
|             settings_json_url = compat_urlparse.urljoin(url, 'includes/settings/%s.json' % trailer_id) | ||||
|             settings_json = self._download_webpage(settings_json_url, trailer_id, u'Downloading settings json') | ||||
|             settings = json.loads(settings_json) | ||||
|             settings = self._download_json(settings_json_url, trailer_id, 'Downloading settings json') | ||||
|  | ||||
|             formats = [] | ||||
|             for format in settings['metadata']['sizes']: | ||||
| @@ -107,7 +109,6 @@ class AppleTrailersIE(InfoExtractor): | ||||
|                 format_url = re.sub(r'_(\d*p.mov)', r'_h\1', format['src']) | ||||
|                 formats.append({ | ||||
|                     'url': format_url, | ||||
|                     'ext': determine_ext(format_url), | ||||
|                     'format': format['type'], | ||||
|                     'width': format['width'], | ||||
|                     'height': int(format['height']), | ||||
|   | ||||
| @@ -38,37 +38,43 @@ class ARDIE(InfoExtractor): | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         title = self._html_search_regex( | ||||
|             r'<h1(?:\s+class="boxTopHeadline")?>(.*?)</h1>', webpage, 'title') | ||||
|             [r'<h1(?:\s+class="boxTopHeadline")?>(.*?)</h1>', | ||||
|              r'<meta name="dcterms.title" content="(.*?)"/>', | ||||
|              r'<h4 class="headline">(.*?)</h4>'], | ||||
|             webpage, 'title') | ||||
|         description = self._html_search_meta( | ||||
|             'dcterms.abstract', webpage, 'description') | ||||
|         thumbnail = self._og_search_thumbnail(webpage) | ||||
|  | ||||
|         streams = [ | ||||
|             mo.groupdict() | ||||
|             for mo in re.finditer( | ||||
|                 r'mediaCollection\.addMediaStream\((?P<media_type>\d+), (?P<quality>\d+), "(?P<rtmp_url>[^"]*)", "(?P<video_url>[^"]*)", "[^"]*"\)', webpage)] | ||||
|  | ||||
|         media_info = self._download_json( | ||||
|             'http://www.ardmediathek.de/play/media/%s' % video_id, video_id) | ||||
|         # The second element of the _mediaArray contains the standard http urls | ||||
|         streams = media_info['_mediaArray'][1]['_mediaStreamArray'] | ||||
|         if not streams: | ||||
|             if '"fsk"' in webpage: | ||||
|                 raise ExtractorError('This video is only available after 20:00') | ||||
|  | ||||
|         formats = [] | ||||
|         for s in streams: | ||||
|             format = { | ||||
|                 'quality': int(s['quality']), | ||||
|             } | ||||
|             if s.get('rtmp_url'): | ||||
|                 format['protocol'] = 'rtmp' | ||||
|                 format['url'] = s['rtmp_url'] | ||||
|                 format['playpath'] = s['video_url'] | ||||
|             else: | ||||
|                 format['url'] = s['video_url'] | ||||
|  | ||||
|             quality_name = self._search_regex( | ||||
|                 r'[,.]([a-zA-Z0-9_-]+),?\.mp4', format['url'], | ||||
|                 'quality name', default='NA') | ||||
|             format['format_id'] = '%s-%s-%s-%s' % ( | ||||
|                 determine_ext(format['url']), quality_name, s['media_type'], | ||||
|                 s['quality']) | ||||
|         for s in streams: | ||||
|             if type(s['_stream']) == list: | ||||
|                 for index, url in enumerate(s['_stream'][::-1]): | ||||
|                     quality = s['_quality'] + index | ||||
|                     formats.append({ | ||||
|                         'quality': quality, | ||||
|                         'url': url, | ||||
|                         'format_id': '%s-%s' % (determine_ext(url), quality) | ||||
|                         }) | ||||
|                 continue | ||||
|  | ||||
|             format = { | ||||
|                 'quality': s['_quality'], | ||||
|                 'url': s['_stream'], | ||||
|             } | ||||
|  | ||||
|             format['format_id'] = '%s-%s' % ( | ||||
|                 determine_ext(format['url']), format['quality']) | ||||
|  | ||||
|             formats.append(format) | ||||
|  | ||||
|   | ||||
| @@ -2,7 +2,6 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import json | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
| @@ -19,114 +18,41 @@ from ..utils import ( | ||||
| # is different for each one. The videos usually expire in 7 days, so we can't | ||||
| # add tests. | ||||
|  | ||||
| class ArteTvIE(InfoExtractor): | ||||
|     _VIDEOS_URL = r'(?:http://)?videos\.arte\.tv/(?P<lang>fr|de)/.*-(?P<id>.*?)\.html' | ||||
|     _LIVEWEB_URL = r'(?:http://)?liveweb\.arte\.tv/(?P<lang>fr|de)/(?P<subpage>.+?)/(?P<name>.+)' | ||||
|     _LIVE_URL = r'index-[0-9]+\.html$' | ||||
|  | ||||
| class ArteTvIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://videos\.arte\.tv/(?P<lang>fr|de)/.*-(?P<id>.*?)\.html' | ||||
|     IE_NAME = 'arte.tv' | ||||
|  | ||||
|     @classmethod | ||||
|     def suitable(cls, url): | ||||
|         return any(re.match(regex, url) for regex in (cls._VIDEOS_URL, cls._LIVEWEB_URL)) | ||||
|  | ||||
|     # TODO implement Live Stream | ||||
|     # from ..utils import compat_urllib_parse | ||||
|     # def extractLiveStream(self, url): | ||||
|     #     video_lang = url.split('/')[-4] | ||||
|     #     info = self.grep_webpage( | ||||
|     #         url, | ||||
|     #         r'src="(.*?/videothek_js.*?\.js)', | ||||
|     #         0, | ||||
|     #         [ | ||||
|     #             (1, 'url', 'Invalid URL: %s' % url) | ||||
|     #         ] | ||||
|     #     ) | ||||
|     #     http_host = url.split('/')[2] | ||||
|     #     next_url = 'http://%s%s' % (http_host, compat_urllib_parse.unquote(info.get('url'))) | ||||
|     #     info = self.grep_webpage( | ||||
|     #         next_url, | ||||
|     #         r'(s_artestras_scst_geoFRDE_' + video_lang + '.*?)\'.*?' + | ||||
|     #             '(http://.*?\.swf).*?' + | ||||
|     #             '(rtmp://.*?)\'', | ||||
|     #         re.DOTALL, | ||||
|     #         [ | ||||
|     #             (1, 'path',   'could not extract video path: %s' % url), | ||||
|     #             (2, 'player', 'could not extract video player: %s' % url), | ||||
|     #             (3, 'url',    'could not extract video url: %s' % url) | ||||
|     #         ] | ||||
|     #     ) | ||||
|     #     video_url = '%s/%s' % (info.get('url'), info.get('path')) | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VIDEOS_URL, url) | ||||
|         if mobj is not None: | ||||
|             id = mobj.group('id') | ||||
|             lang = mobj.group('lang') | ||||
|             return self._extract_video(url, id, lang) | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         lang = mobj.group('lang') | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         mobj = re.match(self._LIVEWEB_URL, url) | ||||
|         if mobj is not None: | ||||
|             name = mobj.group('name') | ||||
|             lang = mobj.group('lang') | ||||
|             return self._extract_liveweb(url, name, lang) | ||||
|  | ||||
|         if re.search(self._LIVE_URL, url) is not None: | ||||
|             raise ExtractorError('Arte live streams are not yet supported, sorry') | ||||
|             # self.extractLiveStream(url) | ||||
|             # return | ||||
|  | ||||
|         raise ExtractorError('No video found') | ||||
|  | ||||
|     def _extract_video(self, url, video_id, lang): | ||||
|         """Extract from videos.arte.tv""" | ||||
|         ref_xml_url = url.replace('/videos/', '/do_delegate/videos/') | ||||
|         ref_xml_url = ref_xml_url.replace('.html', ',view,asPlayerXml.xml') | ||||
|         ref_xml_doc = self._download_xml( | ||||
|             ref_xml_url, video_id, note='Downloading metadata') | ||||
|         config_node = find_xpath_attr(ref_xml_doc, './/video', 'lang', lang) | ||||
|         config_xml_url = config_node.attrib['ref'] | ||||
|         config_xml = self._download_webpage( | ||||
|         config = self._download_xml( | ||||
|             config_xml_url, video_id, note='Downloading configuration') | ||||
|  | ||||
|         video_urls = list(re.finditer(r'<url quality="(?P<quality>.*?)">(?P<url>.*?)</url>', config_xml)) | ||||
|         def _key(m): | ||||
|             quality = m.group('quality') | ||||
|             if quality == 'hd': | ||||
|                 return 2 | ||||
|             else: | ||||
|                 return 1 | ||||
|         # We pick the best quality | ||||
|         video_urls = sorted(video_urls, key=_key) | ||||
|         video_url = list(video_urls)[-1].group('url') | ||||
|          | ||||
|         title = self._html_search_regex(r'<name>(.*?)</name>', config_xml, 'title') | ||||
|         thumbnail = self._html_search_regex(r'<firstThumbnailUrl>(.*?)</firstThumbnailUrl>', | ||||
|                                             config_xml, 'thumbnail') | ||||
|         return {'id': video_id, | ||||
|                 'title': title, | ||||
|                 'thumbnail': thumbnail, | ||||
|                 'url': video_url, | ||||
|                 'ext': 'flv', | ||||
|                 } | ||||
|         formats = [{ | ||||
|             'forma_id': q.attrib['quality'], | ||||
|             'url': q.text, | ||||
|             'ext': 'flv', | ||||
|             'quality': 2 if q.attrib['quality'] == 'hd' else 1, | ||||
|         } for q in config.findall('./urls/url')] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|     def _extract_liveweb(self, url, name, lang): | ||||
|         """Extract form http://liveweb.arte.tv/""" | ||||
|         webpage = self._download_webpage(url, name) | ||||
|         video_id = self._search_regex(r'eventId=(\d+?)("|&)', webpage, 'event id') | ||||
|         config_doc = self._download_xml('http://download.liveweb.arte.tv/o21/liveweb/events/event-%s.xml' % video_id, | ||||
|                                             video_id, 'Downloading information') | ||||
|         event_doc = config_doc.find('event') | ||||
|         url_node = event_doc.find('video').find('urlHd') | ||||
|         if url_node is None: | ||||
|             url_node = event_doc.find('urlSd') | ||||
|  | ||||
|         return {'id': video_id, | ||||
|                 'title': event_doc.find('name%s' % lang.capitalize()).text, | ||||
|                 'url': url_node.text.replace('MP4', 'mp4'), | ||||
|                 'ext': 'flv', | ||||
|                 'thumbnail': self._og_search_thumbnail(webpage), | ||||
|                 } | ||||
|         title = config.find('.//name').text | ||||
|         thumbnail = config.find('.//firstThumbnailUrl').text | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'formats': formats, | ||||
|         } | ||||
|  | ||||
|  | ||||
| class ArteTVPlus7IE(InfoExtractor): | ||||
| @@ -148,13 +74,12 @@ class ArteTVPlus7IE(InfoExtractor): | ||||
|         return self._extract_from_webpage(webpage, video_id, lang) | ||||
|  | ||||
|     def _extract_from_webpage(self, webpage, video_id, lang): | ||||
|         json_url = self._html_search_regex(r'arte_vp_url="(.*?)"', webpage, 'json url') | ||||
|         json_url = self._html_search_regex( | ||||
|             r'arte_vp_url="(.*?)"', webpage, 'json vp url') | ||||
|         return self._extract_from_json_url(json_url, video_id, lang) | ||||
|  | ||||
|     def _extract_from_json_url(self, json_url, video_id, lang): | ||||
|         json_info = self._download_webpage(json_url, video_id, 'Downloading info json') | ||||
|         self.report_extraction(video_id) | ||||
|         info = json.loads(json_info) | ||||
|         info = self._download_json(json_url, video_id) | ||||
|         player_info = info['videoJsonPlayer'] | ||||
|  | ||||
|         info_dict = { | ||||
| @@ -176,6 +101,8 @@ class ArteTVPlus7IE(InfoExtractor): | ||||
|                 l = 'F' | ||||
|             elif lang == 'de': | ||||
|                 l = 'A' | ||||
|             else: | ||||
|                 l = lang | ||||
|             regexes = [r'VO?%s' % l, r'VO?.-ST%s' % l] | ||||
|             return any(re.match(r, f['versionCode']) for r in regexes) | ||||
|         # Some formats may not be in the same language as the url | ||||
| @@ -194,14 +121,17 @@ class ArteTVPlus7IE(InfoExtractor): | ||||
|                 return ['HQ', 'MQ', 'EQ', 'SQ'].index(f['quality']) | ||||
|         else: | ||||
|             def sort_key(f): | ||||
|                 versionCode = f.get('versionCode') | ||||
|                 if versionCode is None: | ||||
|                     versionCode = '' | ||||
|                 return ( | ||||
|                     # Sort first by quality | ||||
|                     int(f.get('height',-1)), | ||||
|                     int(f.get('bitrate',-1)), | ||||
|                     int(f.get('height', -1)), | ||||
|                     int(f.get('bitrate', -1)), | ||||
|                     # The original version with subtitles has lower relevance | ||||
|                     re.match(r'VO-ST(F|A)', f.get('versionCode', '')) is None, | ||||
|                     re.match(r'VO-ST(F|A)', versionCode) is None, | ||||
|                     # The version with sourds/mal subtitles has also lower relevance | ||||
|                     re.match(r'VO?(F|A)-STM\1', f.get('versionCode', '')) is None, | ||||
|                     re.match(r'VO?(F|A)-STM\1', versionCode) is None, | ||||
|                     # Prefer http downloads over m3u8 | ||||
|                     0 if f['url'].endswith('m3u8') else 1, | ||||
|                 ) | ||||
| @@ -305,3 +235,22 @@ class ArteTVConcertIE(ArteTVPlus7IE): | ||||
|             'description': 'md5:486eb08f991552ade77439fe6d82c305', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|  | ||||
| class ArteTVEmbedIE(ArteTVPlus7IE): | ||||
|     IE_NAME = 'arte.tv:embed' | ||||
|     _VALID_URL = r'''(?x) | ||||
|         http://www\.arte\.tv | ||||
|         /playerv2/embed\.php\?json_url= | ||||
|         (?P<json_url> | ||||
|             http://arte\.tv/papi/tvguide/videos/stream/player/ | ||||
|             (?P<lang>[^/]+)/(?P<id>[^/]+)[^&]* | ||||
|         ) | ||||
|     ''' | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         lang = mobj.group('lang') | ||||
|         json_url = mobj.group('json_url') | ||||
|         return self._extract_from_json_url(json_url, video_id, lang) | ||||
|   | ||||
| @@ -11,22 +11,24 @@ from ..utils import ( | ||||
|  | ||||
|  | ||||
| class AUEngineIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://(?:www\.)?auengine\.com/embed\.php\?.*?file=(?P<id>[^&]+).*?' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://auengine.com/embed.php?file=lfvlytY6&w=650&h=370', | ||||
|         'file': 'lfvlytY6.mp4', | ||||
|         'md5': '48972bdbcf1a3a2f5533e62425b41d4f', | ||||
|         'info_dict': { | ||||
|             'id': 'lfvlytY6', | ||||
|             'ext': 'mp4', | ||||
|             'title': '[Commie]The Legend of the Legendary Heroes - 03 - Replication Eye (Alpha Stigma)[F9410F5A]' | ||||
|         } | ||||
|     } | ||||
|     _VALID_URL = r'(?:http://)?(?:www\.)?auengine\.com/embed\.php\?.*?file=([^&]+).*?' | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group(1) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         title = self._html_search_regex(r'<title>(?P<title>.+?)</title>', | ||||
|                 webpage, 'title') | ||||
|         title = self._html_search_regex(r'<title>(?P<title>.+?)</title>', webpage, 'title') | ||||
|         title = title.strip() | ||||
|         links = re.findall(r'\s(?:file|url):\s*["\']([^\'"]+)["\']', webpage) | ||||
|         links = map(compat_urllib_parse.unquote, links) | ||||
| @@ -39,14 +41,15 @@ class AUEngineIE(InfoExtractor): | ||||
|             elif '/videos/' in link: | ||||
|                 video_url = link | ||||
|         if not video_url: | ||||
|             raise ExtractorError(u'Could not find video URL') | ||||
|             raise ExtractorError('Could not find video URL') | ||||
|         ext = '.' + determine_ext(video_url) | ||||
|         if ext == title[-len(ext):]: | ||||
|             title = title[:-len(ext)] | ||||
|  | ||||
|         return { | ||||
|             'id':        video_id, | ||||
|             'url':       video_url, | ||||
|             'title':     title, | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'title': title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'http_referer': 'http://www.auengine.com/flowplayer/flowplayer.commercial-3.2.14.swf', | ||||
|         } | ||||
|   | ||||
| @@ -12,14 +12,14 @@ from ..utils import ( | ||||
|  | ||||
|  | ||||
| class BandcampIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://.*?\.bandcamp\.com/track/(?P<title>.*)' | ||||
|     _VALID_URL = r'https?://.*?\.bandcamp\.com/track/(?P<title>.*)' | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://youtube-dl.bandcamp.com/track/youtube-dl-test-song', | ||||
|         'file': '1812978515.mp3', | ||||
|         'md5': 'c557841d5e50261777a6585648adf439', | ||||
|         'info_dict': { | ||||
|             "title": "youtube-dl  \"'/\\\u00e4\u21ad - youtube-dl test song \"'/\\\u00e4\u21ad", | ||||
|             "duration": 10, | ||||
|             "duration": 9.8485, | ||||
|         }, | ||||
|         '_skip': 'There is a limit of 200 free downloads / month for the test song' | ||||
|     }] | ||||
| @@ -28,36 +28,32 @@ class BandcampIE(InfoExtractor): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         title = mobj.group('title') | ||||
|         webpage = self._download_webpage(url, title) | ||||
|         # We get the link to the free download page | ||||
|         m_download = re.search(r'freeDownloadPage: "(.*?)"', webpage) | ||||
|         if m_download is None: | ||||
|         if not m_download: | ||||
|             m_trackinfo = re.search(r'trackinfo: (.+),\s*?\n', webpage) | ||||
|             if m_trackinfo: | ||||
|                 json_code = m_trackinfo.group(1) | ||||
|                 data = json.loads(json_code) | ||||
|                 d = data[0] | ||||
|                 data = json.loads(json_code)[0] | ||||
|  | ||||
|                 duration = int(round(d['duration'])) | ||||
|                 formats = [] | ||||
|                 for format_id, format_url in d['file'].items(): | ||||
|                     ext, _, abr_str = format_id.partition('-') | ||||
|  | ||||
|                 for format_id, format_url in data['file'].items(): | ||||
|                     ext, abr_str = format_id.split('-', 1) | ||||
|                     formats.append({ | ||||
|                         'format_id': format_id, | ||||
|                         'url': format_url, | ||||
|                         'ext': format_id.partition('-')[0], | ||||
|                         'ext': ext, | ||||
|                         'vcodec': 'none', | ||||
|                         'acodec': format_id.partition('-')[0], | ||||
|                         'abr': int(format_id.partition('-')[2]), | ||||
|                         'acodec': ext, | ||||
|                         'abr': int(abr_str), | ||||
|                     }) | ||||
|  | ||||
|                 self._sort_formats(formats) | ||||
|  | ||||
|                 return { | ||||
|                     'id': compat_str(d['id']), | ||||
|                     'title': d['title'], | ||||
|                     'id': compat_str(data['id']), | ||||
|                     'title': data['title'], | ||||
|                     'formats': formats, | ||||
|                     'duration': duration, | ||||
|                     'duration': float(data['duration']), | ||||
|                 } | ||||
|             else: | ||||
|                 raise ExtractorError('No free songs found') | ||||
| @@ -67,11 +63,9 @@ class BandcampIE(InfoExtractor): | ||||
|             r'var TralbumData = {(.*?)id: (?P<id>\d*?)$', | ||||
|             webpage, re.MULTILINE | re.DOTALL).group('id') | ||||
|  | ||||
|         download_webpage = self._download_webpage(download_link, video_id, | ||||
|                                                   'Downloading free downloads page') | ||||
|         # We get the dictionary of the track from some javascrip code | ||||
|         info = re.search(r'items: (.*?),$', | ||||
|                          download_webpage, re.MULTILINE).group(1) | ||||
|         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] | ||||
|         # We pick mp3-320 for now, until format selection can be easily implemented. | ||||
|         mp3_info = info['downloads']['mp3-320'] | ||||
| @@ -100,7 +94,7 @@ class BandcampIE(InfoExtractor): | ||||
|  | ||||
| class BandcampAlbumIE(InfoExtractor): | ||||
|     IE_NAME = 'Bandcamp:album' | ||||
|     _VALID_URL = r'http://.*?\.bandcamp\.com/album/(?P<title>.*)' | ||||
|     _VALID_URL = r'https?://(?:(?P<subdomain>[^.]+)\.)?bandcamp\.com(?:/album/(?P<title>[^?#]+))' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://blazo.bandcamp.com/album/jazz-format-mixtape-vol-1', | ||||
| @@ -123,13 +117,15 @@ class BandcampAlbumIE(InfoExtractor): | ||||
|         'params': { | ||||
|             'playlistend': 2 | ||||
|         }, | ||||
|         'skip': 'Bancamp imposes download limits. See test_playlists:test_bandcamp_album for the playlist test' | ||||
|         'skip': 'Bandcamp imposes download limits. See test_playlists:test_bandcamp_album for the playlist test' | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         playlist_id = mobj.group('subdomain') | ||||
|         title = mobj.group('title') | ||||
|         webpage = self._download_webpage(url, title) | ||||
|         display_id = title or playlist_id | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|         tracks_paths = re.findall(r'<a href="(.*?)" itemprop="url">', webpage) | ||||
|         if not tracks_paths: | ||||
|             raise ExtractorError('The page doesn\'t contain any tracks') | ||||
| @@ -139,6 +135,8 @@ class BandcampAlbumIE(InfoExtractor): | ||||
|         title = self._search_regex(r'album_title : "(.*?)"', webpage, 'title') | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|             'id': playlist_id, | ||||
|             'display_id': display_id, | ||||
|             'title': title, | ||||
|             'entries': entries, | ||||
|         } | ||||
|   | ||||
							
								
								
									
										106
									
								
								youtube_dl/extractor/bilibili.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								youtube_dl/extractor/bilibili.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,106 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_parse_qs, | ||||
|     ExtractorError, | ||||
|     int_or_none, | ||||
|     unified_strdate, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BiliBiliIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://www\.bilibili\.(?:tv|com)/video/av(?P<id>[0-9]+)/' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.bilibili.tv/video/av1074402/', | ||||
|         'md5': '2c301e4dab317596e837c3e7633e7d86', | ||||
|         'info_dict': { | ||||
|             'id': '1074402', | ||||
|             'ext': 'flv', | ||||
|             'title': '【金坷垃】金泡沫', | ||||
|             'duration': 308, | ||||
|             'upload_date': '20140420', | ||||
|             'thumbnail': 're:^https?://.+\.jpg', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         video_code = self._search_regex( | ||||
|             r'(?s)<div itemprop="video".*?>(.*?)</div>', webpage, 'video code') | ||||
|  | ||||
|         title = self._html_search_meta( | ||||
|             'media:title', video_code, 'title', fatal=True) | ||||
|         duration_str = self._html_search_meta( | ||||
|             'duration', video_code, 'duration') | ||||
|         if duration_str is None: | ||||
|             duration = None | ||||
|         else: | ||||
|             duration_mobj = re.match( | ||||
|                 r'^T(?:(?P<hours>[0-9]+)H)?(?P<minutes>[0-9]+)M(?P<seconds>[0-9]+)S$', | ||||
|                 duration_str) | ||||
|             duration = ( | ||||
|                 int_or_none(duration_mobj.group('hours'), default=0) * 3600 + | ||||
|                 int(duration_mobj.group('minutes')) * 60 + | ||||
|                 int(duration_mobj.group('seconds'))) | ||||
|         upload_date = unified_strdate(self._html_search_meta( | ||||
|             'uploadDate', video_code, fatal=False)) | ||||
|         thumbnail = self._html_search_meta( | ||||
|             'thumbnailUrl', video_code, 'thumbnail', fatal=False) | ||||
|  | ||||
|         player_params = compat_parse_qs(self._html_search_regex( | ||||
|             r'<iframe .*?class="player" src="https://secure\.bilibili\.(?:tv|com)/secure,([^"]+)"', | ||||
|             webpage, 'player params')) | ||||
|  | ||||
|         if 'cid' in player_params: | ||||
|             cid = player_params['cid'][0] | ||||
|  | ||||
|             lq_doc = self._download_xml( | ||||
|                 'http://interface.bilibili.cn/v_cdn_play?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'), | ||||
|             }] | ||||
|  | ||||
|             hq_doc = self._download_xml( | ||||
|                 'http://interface.bilibili.cn/playurl?cid=%s' % cid, | ||||
|                 video_id, | ||||
|                 note='Downloading HQ video info', | ||||
|                 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, | ||||
|                     'filesize': int_or_none( | ||||
|                         hq_durl.find('./size'), get_attr='text'), | ||||
|                 }) | ||||
|         else: | ||||
|             raise ExtractorError('Unsupported player parameters: %r' % (player_params,)) | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'formats': formats, | ||||
|             'duration': duration, | ||||
|             'upload_date': upload_date, | ||||
|             'thumbnail': thumbnail, | ||||
|         } | ||||
| @@ -1,13 +1,10 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import datetime | ||||
| import json | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     remove_start, | ||||
| ) | ||||
| from ..utils import remove_start | ||||
|  | ||||
|  | ||||
| class BlinkxIE(InfoExtractor): | ||||
| @@ -16,18 +13,21 @@ class BlinkxIE(InfoExtractor): | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.blinkx.com/ce/8aQUy7GVFYgFzpKhT0oqsilwOGFRVXk3R1ZGWWdGenBLaFQwb3FzaWx3OGFRVXk3R1ZGWWdGenB', | ||||
|         'file': '8aQUy7GV.mp4', | ||||
|         'md5': '2e9a07364af40163a908edbf10bb2492', | ||||
|         'info_dict': { | ||||
|             "title": "Police Car Rolls Away", | ||||
|             "uploader": "stupidvideos.com", | ||||
|             "upload_date": "20131215", | ||||
|             "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, | ||||
|                 "url": "http://cdn.blinkx.com/stream/b/41/StupidVideos/20131215/1873969261/1873969261_tn_0.jpg", | ||||
|             'id': '8aQUy7GV', | ||||
|             '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', | ||||
|             }], | ||||
|         }, | ||||
|     } | ||||
| @@ -37,13 +37,10 @@ class BlinkxIE(InfoExtractor): | ||||
|         video_id = m.group('id') | ||||
|         display_id = video_id[:8] | ||||
|  | ||||
|         api_url = (u'https://apib4.blinkx.com/api.php?action=play_video&' + | ||||
|         api_url = ('https://apib4.blinkx.com/api.php?action=play_video&' + | ||||
|                    'video=%s' % video_id) | ||||
|         data_json = self._download_webpage(api_url, display_id) | ||||
|         data = json.loads(data_json)['api']['results'][0] | ||||
|         dt = datetime.datetime.fromtimestamp(data['pubdate_epoch']) | ||||
|         pload_date = dt.strftime('%Y%m%d') | ||||
|  | ||||
|         duration = None | ||||
|         thumbnails = [] | ||||
|         formats = [] | ||||
| @@ -58,16 +55,13 @@ class BlinkxIE(InfoExtractor): | ||||
|                 duration = m['d'] | ||||
|             elif m['type'] == 'youtube': | ||||
|                 yt_id = m['link'] | ||||
|                 self.to_screen(u'Youtube video detected: %s' % yt_id) | ||||
|                 self.to_screen('Youtube video detected: %s' % yt_id) | ||||
|                 return self.url_result(yt_id, 'Youtube', video_id=yt_id) | ||||
|             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 | ||||
|                 format_id = (u'%s-%sk-%s' % | ||||
|                              (vcodec, | ||||
|                               tbr, | ||||
|                               m['w'])) | ||||
|                 format_id = '%s-%sk-%s' % (vcodec, tbr, m['w']) | ||||
|                 formats.append({ | ||||
|                     'format_id': format_id, | ||||
|                     'url': m['link'], | ||||
| @@ -88,7 +82,7 @@ class BlinkxIE(InfoExtractor): | ||||
|             'title': data['title'], | ||||
|             'formats': formats, | ||||
|             'uploader': data['channel_name'], | ||||
|             'upload_date': pload_date, | ||||
|             'timestamp': data['pubdate_epoch'], | ||||
|             'description': data.get('description'), | ||||
|             'thumbnails': thumbnails, | ||||
|             'duration': duration, | ||||
|   | ||||
| @@ -1,102 +1,124 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import datetime | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from .subtitles import SubtitlesInfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_str, | ||||
|     compat_urllib_request, | ||||
|  | ||||
|     unescapeHTML, | ||||
|     parse_iso8601, | ||||
|     compat_urlparse, | ||||
|     clean_html, | ||||
|     compat_str, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BlipTVIE(SubtitlesInfoExtractor): | ||||
|     """Information extractor for blip.tv""" | ||||
|     _VALID_URL = r'https?://(?:\w+\.)?blip\.tv/(?:(?:.+-|rss/flash/)(?P<id>\d+)|((?:play/|api\.swf#)(?P<lookup_id>[\da-zA-Z+]+)))' | ||||
|  | ||||
|     _VALID_URL = r'https?://(?:\w+\.)?blip\.tv/((.+/)|(play/)|(api\.swf#))(?P<presumptive_id>.+)$' | ||||
|  | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://blip.tv/cbr/cbr-exclusive-gotham-city-imposters-bats-vs-jokerz-short-3-5796352', | ||||
|         'md5': 'c6934ad0b6acf2bd920720ec888eb812', | ||||
|         'info_dict': { | ||||
|             'id': '5779306', | ||||
|             'ext': 'mov', | ||||
|             'upload_date': '20111205', | ||||
|             'description': 'md5:9bc31f227219cde65e47eeec8d2dc596', | ||||
|             'uploader': 'Comic Book Resources - CBR TV', | ||||
|             'title': 'CBR EXCLUSIVE: "Gotham City Imposters" Bats VS Jokerz Short 3', | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             'url': 'http://blip.tv/cbr/cbr-exclusive-gotham-city-imposters-bats-vs-jokerz-short-3-5796352', | ||||
|             'md5': 'c6934ad0b6acf2bd920720ec888eb812', | ||||
|             'info_dict': { | ||||
|                 'id': '5779306', | ||||
|                 'ext': 'mov', | ||||
|                 'title': 'CBR EXCLUSIVE: "Gotham City Imposters" Bats VS Jokerz Short 3', | ||||
|                 'description': 'md5:9bc31f227219cde65e47eeec8d2dc596', | ||||
|                 'timestamp': 1323138843, | ||||
|                 'upload_date': '20111206', | ||||
|                 'uploader': 'cbr', | ||||
|                 'uploader_id': '679425', | ||||
|                 'duration': 81, | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             # https://github.com/rg3/youtube-dl/pull/2274 | ||||
|             'note': 'Video with subtitles', | ||||
|             'url': 'http://blip.tv/play/h6Uag5OEVgI.html', | ||||
|             'md5': '309f9d25b820b086ca163ffac8031806', | ||||
|             'info_dict': { | ||||
|                 'id': '6586561', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Red vs. Blue Season 11 Episode 1', | ||||
|                 'description': 'One-Zero-One', | ||||
|                 'timestamp': 1371261608, | ||||
|                 'upload_date': '20130615', | ||||
|                 'uploader': 'redvsblue', | ||||
|                 'uploader_id': '792887', | ||||
|                 'duration': 279, | ||||
|             } | ||||
|         } | ||||
|     }, { | ||||
|         # https://github.com/rg3/youtube-dl/pull/2274 | ||||
|         'note': 'Video with subtitles', | ||||
|         'url': 'http://blip.tv/play/h6Uag5OEVgI.html', | ||||
|         'md5': '309f9d25b820b086ca163ffac8031806', | ||||
|         'info_dict': { | ||||
|             'id': '6586561', | ||||
|             'ext': 'mp4', | ||||
|             'uploader': 'Red vs. Blue', | ||||
|             'description': 'One-Zero-One', | ||||
|             'upload_date': '20130614', | ||||
|             'title': 'Red vs. Blue Season 11 Episode 1', | ||||
|         } | ||||
|     }] | ||||
|     ] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         presumptive_id = mobj.group('presumptive_id') | ||||
|         lookup_id = mobj.group('lookup_id') | ||||
|  | ||||
|         # See https://github.com/rg3/youtube-dl/issues/857 | ||||
|         embed_mobj = re.match(r'https?://(?:\w+\.)?blip\.tv/(?:play/|api\.swf#)([a-zA-Z0-9]+)', url) | ||||
|         if embed_mobj: | ||||
|             info_url = 'http://blip.tv/play/%s.x?p=1' % embed_mobj.group(1) | ||||
|             info_page = self._download_webpage(info_url, embed_mobj.group(1)) | ||||
|             video_id = self._search_regex( | ||||
|                 r'data-episode-id="([0-9]+)', info_page, 'video_id') | ||||
|             return self.url_result('http://blip.tv/a/a-' + video_id, 'BlipTV') | ||||
|          | ||||
|         cchar = '&' if '?' in url else '?' | ||||
|         json_url = url + cchar + 'skin=json&version=2&no_wrap=1' | ||||
|         request = compat_urllib_request.Request(json_url) | ||||
|         request.add_header('User-Agent', 'iTunes/10.6.1') | ||||
|  | ||||
|         json_data = self._download_json(request, video_id=presumptive_id) | ||||
|  | ||||
|         if 'Post' in json_data: | ||||
|             data = json_data['Post'] | ||||
|         if lookup_id: | ||||
|             info_page = self._download_webpage( | ||||
|                 'http://blip.tv/play/%s.x?p=1' % lookup_id, lookup_id, 'Resolving lookup id') | ||||
|             video_id = self._search_regex(r'data-episode-id="([0-9]+)', info_page, 'video_id') | ||||
|         else: | ||||
|             data = json_data | ||||
|             video_id = mobj.group('id') | ||||
|  | ||||
|         rss = self._download_xml('http://blip.tv/rss/flash/%s' % video_id, video_id, 'Downloading video RSS') | ||||
|  | ||||
|         def blip(s): | ||||
|             return '{http://blip.tv/dtd/blip/1.0}%s' % s | ||||
|  | ||||
|         def media(s): | ||||
|             return '{http://search.yahoo.com/mrss/}%s' % s | ||||
|  | ||||
|         def itunes(s): | ||||
|             return '{http://www.itunes.com/dtds/podcast-1.0.dtd}%s' % s | ||||
|  | ||||
|         item = rss.find('channel/item') | ||||
|  | ||||
|         video_id = item.find(blip('item_id')).text | ||||
|         title = item.find('./title').text | ||||
|         description = clean_html(compat_str(item.find(blip('puredescription')).text)) | ||||
|         timestamp = parse_iso8601(item.find(blip('datestamp')).text) | ||||
|         uploader = item.find(blip('user')).text | ||||
|         uploader_id = item.find(blip('userid')).text | ||||
|         duration = int(item.find(blip('runtime')).text) | ||||
|         media_thumbnail = item.find(media('thumbnail')) | ||||
|         thumbnail = media_thumbnail.get('url') if media_thumbnail is not None else item.find(itunes('image')).text | ||||
|         categories = [category.text for category in item.findall('category')] | ||||
|  | ||||
|         video_id = compat_str(data['item_id']) | ||||
|         upload_date = datetime.datetime.strptime(data['datestamp'], '%m-%d-%y %H:%M%p').strftime('%Y%m%d') | ||||
|         subtitles = {} | ||||
|         formats = [] | ||||
|         if 'additionalMedia' in data: | ||||
|             for f in data['additionalMedia']: | ||||
|                 if f.get('file_type_srt') == 1: | ||||
|                     LANGS = { | ||||
|                         'english': 'en', | ||||
|                     } | ||||
|                     lang = f['role'].rpartition('-')[-1].strip().lower() | ||||
|                     langcode = LANGS.get(lang, lang) | ||||
|                     subtitles[langcode] = f['url'] | ||||
|                     continue | ||||
|                 if not int(f['media_width']):  # filter m3u8 | ||||
|                     continue | ||||
|         subtitles = {} | ||||
|  | ||||
|         media_group = item.find(media('group')) | ||||
|         for media_content in media_group.findall(media('content')): | ||||
|             url = media_content.get('url') | ||||
|             role = media_content.get(blip('role')) | ||||
|             msg = self._download_webpage( | ||||
|                 url + '?showplayer=20140425131715&referrer=http://blip.tv&mask=7&skin=flashvars&view=url', | ||||
|                 video_id, 'Resolving URL for %s' % role) | ||||
|             real_url = compat_urlparse.parse_qs(msg)['message'][0] | ||||
|  | ||||
|             media_type = media_content.get('type') | ||||
|             if media_type == 'text/srt' or url.endswith('.srt'): | ||||
|                 LANGS = { | ||||
|                     'english': 'en', | ||||
|                 } | ||||
|                 lang = role.rpartition('-')[-1].strip().lower() | ||||
|                 langcode = LANGS.get(lang, lang) | ||||
|                 subtitles[langcode] = url | ||||
|             elif media_type.startswith('video/'): | ||||
|                 formats.append({ | ||||
|                     'url': f['url'], | ||||
|                     'format_id': f['role'], | ||||
|                     'width': int(f['media_width']), | ||||
|                     'height': int(f['media_height']), | ||||
|                     'url': real_url, | ||||
|                     'format_id': role, | ||||
|                     'format_note': media_type, | ||||
|                     'vcodec': media_content.get(blip('vcodec')), | ||||
|                     'acodec': media_content.get(blip('acodec')), | ||||
|                     'filesize': media_content.get('filesize'), | ||||
|                     'width': int(media_content.get('width')), | ||||
|                     'height': int(media_content.get('height')), | ||||
|                 }) | ||||
|         else: | ||||
|             formats.append({ | ||||
|                 'url': data['media']['url'], | ||||
|                 'width': int(data['media']['width']), | ||||
|                 'height': int(data['media']['height']), | ||||
|             }) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         # subtitles | ||||
| @@ -107,12 +129,14 @@ class BlipTVIE(SubtitlesInfoExtractor): | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'uploader': data['display_name'], | ||||
|             'upload_date': upload_date, | ||||
|             'title': data['title'], | ||||
|             'thumbnail': data['thumbnailUrl'], | ||||
|             'description': data['description'], | ||||
|             'user_agent': 'iTunes/10.6.1', | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'timestamp': timestamp, | ||||
|             'uploader': uploader, | ||||
|             'uploader_id': uploader_id, | ||||
|             'duration': duration, | ||||
|             'thumbnail': thumbnail, | ||||
|             'categories': categories, | ||||
|             'formats': formats, | ||||
|             'subtitles': video_subtitles, | ||||
|         } | ||||
|   | ||||
| @@ -1,22 +1,21 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from .ooyala import OoyalaIE | ||||
|  | ||||
|  | ||||
| class BloombergIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://www\.bloomberg\.com/video/(?P<name>.+?)\.html' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.bloomberg.com/video/shah-s-presentation-on-foreign-exchange-strategies-qurhIVlJSB6hzkVi229d8g.html', | ||||
|         u'file': u'12bzhqZTqQHmmlA8I-i0NpzJgcG5NNYX.mp4', | ||||
|         u'info_dict': { | ||||
|             u'title': u'Shah\'s Presentation on Foreign-Exchange Strategies', | ||||
|             u'description': u'md5:abc86e5236f9f0e4866c59ad36736686', | ||||
|         }, | ||||
|         u'params': { | ||||
|             # Requires ffmpeg (m3u8 manifest) | ||||
|             u'skip_download': True, | ||||
|         'url': 'http://www.bloomberg.com/video/shah-s-presentation-on-foreign-exchange-strategies-qurhIVlJSB6hzkVi229d8g.html', | ||||
|         'md5': '7bf08858ff7c203c870e8a6190e221e5', | ||||
|         'info_dict': { | ||||
|             'id': 'qurhIVlJSB6hzkVi229d8g', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Shah\'s Presentation on Foreign-Exchange Strategies', | ||||
|             'description': 'md5:0681e0d30dcdfc6abf34594961d8ea88', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
| @@ -24,7 +23,16 @@ class BloombergIE(InfoExtractor): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         name = mobj.group('name') | ||||
|         webpage = self._download_webpage(url, name) | ||||
|         embed_code = self._search_regex( | ||||
|             r'<source src="https?://[^/]+/[^/]+/[^/]+/([^/]+)', webpage, | ||||
|             'embed code') | ||||
|         return OoyalaIE._build_url_result(embed_code) | ||||
|         f4m_url = self._search_regex( | ||||
|             r'<source src="(https?://[^"]+\.f4m.*?)"', webpage, | ||||
|             'f4m url') | ||||
|         title = re.sub(': Video$', '', self._og_search_title(webpage)) | ||||
|  | ||||
|         return { | ||||
|             'id': name.split('-')[-1], | ||||
|             'title': title, | ||||
|             'url': f4m_url, | ||||
|             'ext': 'flv', | ||||
|             'description': self._og_search_description(webpage), | ||||
|             'thumbnail': self._og_search_thumbnail(webpage), | ||||
|         } | ||||
|   | ||||
| @@ -4,39 +4,70 @@ from __future__ import unicode_literals | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ExtractorError | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     int_or_none, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BRIE(InfoExtractor): | ||||
|     IE_DESC = "Bayerischer Rundfunk Mediathek" | ||||
|     _VALID_URL = r"^https?://(?:www\.)?br\.de/mediathek/video/(?:sendungen/)?(?:[a-z0-9\-/]+/)?(?P<id>[a-z0-9\-]+)\.html$" | ||||
|     _BASE_URL = "http://www.br.de" | ||||
|     IE_DESC = 'Bayerischer Rundfunk Mediathek' | ||||
|     _VALID_URL = r'https?://(?:www\.)?br\.de/(?:[a-z0-9\-]+/)+(?P<id>[a-z0-9\-]+)\.html' | ||||
|     _BASE_URL = 'http://www.br.de' | ||||
|  | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             "url": "http://www.br.de/mediathek/video/anselm-gruen-114.html", | ||||
|             "md5": "c4f83cf0f023ba5875aba0bf46860df2", | ||||
|             "info_dict": { | ||||
|                 "id": "2c8d81c5-6fb7-4a74-88d4-e768e5856532", | ||||
|                 "ext": "mp4", | ||||
|                 "title": "Feiern und Verzichten", | ||||
|                 "description": "Anselm Grün: Feiern und Verzichten", | ||||
|                 "uploader": "BR/Birgit Baier", | ||||
|                 "upload_date": "20140301" | ||||
|             'url': 'http://www.br.de/mediathek/video/sendungen/heimatsound/heimatsound-festival-2014-trailer-100.html', | ||||
|             'md5': '93556dd2bcb2948d9259f8670c516d59', | ||||
|             'info_dict': { | ||||
|                 'id': '25e279aa-1ffd-40fd-9955-5325bd48a53a', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Am 1. und 2. August in Oberammergau', | ||||
|                 'description': 'md5:dfd224e5aa6819bc1fcbb7826a932021', | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             "url": "http://www.br.de/mediathek/video/sendungen/unter-unserem-himmel/unter-unserem-himmel-alpen-ueber-den-pass-100.html", | ||||
|             "md5": "ab451b09d861dbed7d7cc9ab0be19ebe", | ||||
|             "info_dict": { | ||||
|                 "id": "2c060e69-3a27-4e13-b0f0-668fac17d812", | ||||
|                 "ext": "mp4", | ||||
|                 "title": "Über den Pass", | ||||
|                 "description": "Die Eroberung der Alpen: Über den Pass", | ||||
|                 "uploader": None, | ||||
|                 "upload_date": None | ||||
|             'url': 'http://www.br.de/mediathek/video/sendungen/unter-unserem-himmel/unter-unserem-himmel-alpen-ueber-den-pass-100.html', | ||||
|             'md5': 'ab451b09d861dbed7d7cc9ab0be19ebe', | ||||
|             'info_dict': { | ||||
|                 'id': '2c060e69-3a27-4e13-b0f0-668fac17d812', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Über den Pass', | ||||
|                 'description': 'Die Eroberung der Alpen: Über den Pass', | ||||
|             } | ||||
|         } | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://www.br.de/nachrichten/schaeuble-haushaltsentwurf-bundestag-100.html', | ||||
|             'md5': '3db0df1a9a9cd9fa0c70e6ea8aa8e820', | ||||
|             'info_dict': { | ||||
|                 'id': 'c6aae3de-2cf9-43f2-957f-f17fef9afaab', | ||||
|                 'ext': 'aac', | ||||
|                 'title': '"Keine neuen Schulden im nächsten Jahr"', | ||||
|                 'description': 'Haushaltsentwurf: "Keine neuen Schulden im nächsten Jahr"', | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://www.br.de/radio/bayern1/service/team/videos/team-video-erdelt100.html', | ||||
|             'md5': 'dbab0aef2e047060ea7a21fc1ce1078a', | ||||
|             'info_dict': { | ||||
|                 'id': '6ba73750-d405-45d3-861d-1ce8c524e059', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Umweltbewusster Häuslebauer', | ||||
|                 'description': 'Uwe Erdelt: Umweltbewusster Häuslebauer', | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://www.br.de/fernsehen/br-alpha/sendungen/kant-fuer-anfaenger/kritik-der-reinen-vernunft/kant-kritik-01-metaphysik100.html', | ||||
|             'md5': '23bca295f1650d698f94fc570977dae3', | ||||
|             'info_dict': { | ||||
|                 'id': 'd982c9ce-8648-4753-b358-98abb8aec43d', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Folge 1 - Metaphysik', | ||||
|                 'description': 'Kant für Anfänger: Folge 1 - Metaphysik', | ||||
|                 'uploader': 'Eva Maria Steimle', | ||||
|                 'upload_date': '20140117', | ||||
|             } | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
| @@ -44,56 +75,63 @@ class BRIE(InfoExtractor): | ||||
|         display_id = mobj.group('id') | ||||
|         page = self._download_webpage(url, display_id) | ||||
|         xml_url = self._search_regex( | ||||
|             r"return BRavFramework\.register\(BRavFramework\('avPlayer_(?:[a-f0-9-]{36})'\)\.setup\({dataURL:'(/mediathek/video/[a-z0-9/~_.-]+)'}\)\);", page, "XMLURL") | ||||
|             r"return BRavFramework\.register\(BRavFramework\('avPlayer_(?:[a-f0-9-]{36})'\)\.setup\({dataURL:'(/(?:[a-z0-9\-]+/)+[a-z0-9/~_.-]+)'}\)\);", page, 'XMLURL') | ||||
|         xml = self._download_xml(self._BASE_URL + xml_url, None) | ||||
|  | ||||
|         videos = [] | ||||
|         for xml_video in xml.findall("video"): | ||||
|             video = { | ||||
|                 "id": xml_video.get("externalId"), | ||||
|                 "title": xml_video.find("title").text, | ||||
|                 "formats": self._extract_formats(xml_video.find("assets")), | ||||
|                 "thumbnails": self._extract_thumbnails(xml_video.find("teaserImage/variants")), | ||||
|                 "description": " ".join(xml_video.find("shareTitle").text.splitlines()), | ||||
|                 "webpage_url": xml_video.find("permalink").text | ||||
|             } | ||||
|             if xml_video.find("author").text: | ||||
|                 video["uploader"] = xml_video.find("author").text | ||||
|             if xml_video.find("broadcastDate").text: | ||||
|                 video["upload_date"] =  "".join(reversed(xml_video.find("broadcastDate").text.split("."))) | ||||
|             videos.append(video) | ||||
|         medias = [] | ||||
|  | ||||
|         if len(videos) > 1: | ||||
|         for xml_media in xml.findall('video') + xml.findall('audio'): | ||||
|             media = { | ||||
|                 'id': xml_media.get('externalId'), | ||||
|                 'title': xml_media.find('title').text, | ||||
|                 'formats': self._extract_formats(xml_media.find('assets')), | ||||
|                 'thumbnails': self._extract_thumbnails(xml_media.find('teaserImage/variants')), | ||||
|                 'description': ' '.join(xml_media.find('shareTitle').text.splitlines()), | ||||
|                 'webpage_url': xml_media.find('permalink').text | ||||
|             } | ||||
|             if xml_media.find('author').text: | ||||
|                 media['uploader'] = xml_media.find('author').text | ||||
|             if xml_media.find('broadcastDate').text: | ||||
|                 media['upload_date'] = ''.join(reversed(xml_media.find('broadcastDate').text.split('.'))) | ||||
|             medias.append(media) | ||||
|  | ||||
|         if len(medias) > 1: | ||||
|             self._downloader.report_warning( | ||||
|                 'found multiple videos; please ' | ||||
|                 'found multiple medias; please ' | ||||
|                 'report this with the video URL to http://yt-dl.org/bug') | ||||
|         if not videos: | ||||
|             raise ExtractorError('No video entries found') | ||||
|         return videos[0] | ||||
|         if not medias: | ||||
|             raise ExtractorError('No media entries found') | ||||
|         return medias[0] | ||||
|  | ||||
|     def _extract_formats(self, assets): | ||||
|  | ||||
|         def text_or_none(asset, tag): | ||||
|             elem = asset.find(tag) | ||||
|             return None if elem is None else elem.text | ||||
|  | ||||
|         formats = [{ | ||||
|             "url": asset.find("downloadUrl").text, | ||||
|             "ext": asset.find("mediaType").text, | ||||
|             "format_id": asset.get("type"), | ||||
|             "width": int(asset.find("frameWidth").text), | ||||
|             "height": int(asset.find("frameHeight").text), | ||||
|             "tbr": int(asset.find("bitrateVideo").text), | ||||
|             "abr": int(asset.find("bitrateAudio").text), | ||||
|             "vcodec": asset.find("codecVideo").text, | ||||
|             "container": asset.find("mediaType").text, | ||||
|             "filesize": int(asset.find("size").text), | ||||
|         } for asset in assets.findall("asset") | ||||
|             if asset.find("downloadUrl") is not None] | ||||
|             'url': text_or_none(asset, 'downloadUrl'), | ||||
|             'ext': text_or_none(asset, 'mediaType'), | ||||
|             'format_id': asset.get('type'), | ||||
|             'width': int_or_none(text_or_none(asset, 'frameWidth')), | ||||
|             'height': int_or_none(text_or_none(asset, 'frameHeight')), | ||||
|             'tbr': int_or_none(text_or_none(asset, 'bitrateVideo')), | ||||
|             'abr': int_or_none(text_or_none(asset, 'bitrateAudio')), | ||||
|             'vcodec': text_or_none(asset, 'codecVideo'), | ||||
|             'acodec': text_or_none(asset, 'codecAudio'), | ||||
|             'container': text_or_none(asset, 'mediaType'), | ||||
|             'filesize': int_or_none(text_or_none(asset, 'size')), | ||||
|         } for asset in assets.findall('asset') | ||||
|             if asset.find('downloadUrl') is not None] | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|         return formats | ||||
|  | ||||
|     def _extract_thumbnails(self, variants): | ||||
|         thumbnails = [{ | ||||
|             "url": self._BASE_URL + variant.find("url").text, | ||||
|             "width": int(variant.find("width").text), | ||||
|             "height": int(variant.find("height").text), | ||||
|         } for variant in variants.findall("variant")] | ||||
|         thumbnails.sort(key=lambda x: x["width"] * x["height"], reverse=True) | ||||
|             'url': self._BASE_URL + variant.find('url').text, | ||||
|             'width': int_or_none(variant.find('width').text), | ||||
|             'height': int_or_none(variant.find('height').text), | ||||
|         } for variant in variants.findall('variant')] | ||||
|         thumbnails.sort(key=lambda x: x['width'] * x['height'], reverse=True) | ||||
|         return thumbnails | ||||
|   | ||||
| @@ -27,9 +27,10 @@ class BreakIE(InfoExtractor): | ||||
|             webpage, 'info json', flags=re.DOTALL) | ||||
|         info = json.loads(info_json) | ||||
|         video_url = info['videoUri'] | ||||
|         m_youtube = re.search(r'(https?://www\.youtube\.com/watch\?v=.*)', video_url) | ||||
|         if m_youtube is not None: | ||||
|             return self.url_result(m_youtube.group(1), 'Youtube') | ||||
|         youtube_id = info.get('youtubeId') | ||||
|         if youtube_id: | ||||
|             return self.url_result(youtube_id, 'Youtube') | ||||
|  | ||||
|         final_url = video_url + '?' + info['AuthToken'] | ||||
|         return { | ||||
|             'id': video_id, | ||||
|   | ||||
| @@ -15,6 +15,7 @@ from ..utils import ( | ||||
|     compat_urllib_request, | ||||
|     compat_parse_qs, | ||||
|  | ||||
|     determine_ext, | ||||
|     ExtractorError, | ||||
|     unsmuggle_url, | ||||
|     unescapeHTML, | ||||
| @@ -29,10 +30,11 @@ class BrightcoveIE(InfoExtractor): | ||||
|         { | ||||
|             # From http://www.8tv.cat/8aldia/videos/xavier-sala-i-martin-aquesta-tarda-a-8-al-dia/ | ||||
|             'url': 'http://c.brightcove.com/services/viewer/htmlFederated?playerID=1654948606001&flashID=myExperience&%40videoPlayer=2371591881001', | ||||
|             'file': '2371591881001.mp4', | ||||
|             'md5': '5423e113865d26e40624dce2e4b45d95', | ||||
|             'note': 'Test Brightcove downloads and detection in GenericIE', | ||||
|             'info_dict': { | ||||
|                 'id': '2371591881001', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Xavier Sala i Martín: “Un banc que no presta és un banc zombi que no serveix per a res”', | ||||
|                 'uploader': '8TV', | ||||
|                 'description': 'md5:a950cc4285c43e44d763d036710cd9cd', | ||||
| @@ -41,8 +43,9 @@ class BrightcoveIE(InfoExtractor): | ||||
|         { | ||||
|             # From http://medianetwork.oracle.com/video/player/1785452137001 | ||||
|             'url': 'http://c.brightcove.com/services/viewer/htmlFederated?playerID=1217746023001&flashID=myPlayer&%40videoPlayer=1785452137001', | ||||
|             'file': '1785452137001.flv', | ||||
|             'info_dict': { | ||||
|                 'id': '1785452137001', | ||||
|                 'ext': 'flv', | ||||
|                 'title': 'JVMLS 2012: Arrays 2.0 - Opportunities and Challenges', | ||||
|                 'description': 'John Rose speaks at the JVM Language Summit, August 1, 2012.', | ||||
|                 'uploader': 'Oracle', | ||||
| @@ -70,7 +73,20 @@ class BrightcoveIE(InfoExtractor): | ||||
|                 'description': 'md5:363109c02998fee92ec02211bd8000df', | ||||
|                 'uploader': 'National Ballet of Canada', | ||||
|             }, | ||||
|         } | ||||
|         }, | ||||
|         { | ||||
|             # test flv videos served by akamaihd.net | ||||
|             # From http://www.redbull.com/en/bike/stories/1331655643987/replay-uci-dh-world-cup-2014-from-fort-william | ||||
|             'url': 'http://c.brightcove.com/services/viewer/htmlFederated?%40videoPlayer=ref%3ABC2996102916001&linkBaseURL=http%3A%2F%2Fwww.redbull.com%2Fen%2Fbike%2Fvideos%2F1331655630249%2Freplay-uci-fort-william-2014-dh&playerKey=AQ%7E%7E%2CAAAApYJ7UqE%7E%2Cxqr_zXk0I-zzNndy8NlHogrCb5QdyZRf&playerID=1398061561001#__youtubedl_smuggle=%7B%22Referer%22%3A+%22http%3A%2F%2Fwww.redbull.com%2Fen%2Fbike%2Fstories%2F1331655643987%2Freplay-uci-dh-world-cup-2014-from-fort-william%22%7D', | ||||
|             # The md5 checksum changes on each download | ||||
|             'info_dict': { | ||||
|                 'id': '2996102916001', | ||||
|                 'ext': 'flv', | ||||
|                 'title': 'UCI MTB World Cup 2014: Fort William, UK - Downhill Finals', | ||||
|                 'uploader': 'Red Bull TV', | ||||
|                 'description': 'UCI MTB World Cup 2014: Fort William, UK - Downhill Finals', | ||||
|             }, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     @classmethod | ||||
| @@ -87,7 +103,7 @@ class BrightcoveIE(InfoExtractor): | ||||
|         object_str = object_str.replace('<--', '<!--') | ||||
|         object_str = fix_xml_ampersands(object_str) | ||||
|  | ||||
|         object_doc = xml.etree.ElementTree.fromstring(object_str) | ||||
|         object_doc = xml.etree.ElementTree.fromstring(object_str.encode('utf-8')) | ||||
|  | ||||
|         fv_el = find_xpath_attr(object_doc, './param', 'name', 'flashVars') | ||||
|         if fv_el is not None: | ||||
| @@ -140,7 +156,11 @@ class BrightcoveIE(InfoExtractor): | ||||
|  | ||||
|         url_m = re.search(r'<meta\s+property="og:video"\s+content="(http://c.brightcove.com/[^"]+)"', webpage) | ||||
|         if url_m: | ||||
|             return [unescapeHTML(url_m.group(1))] | ||||
|             url = unescapeHTML(url_m.group(1)) | ||||
|             # Some sites don't add it, we can't download with this url, for example: | ||||
|             # http://www.ktvu.com/videos/news/raw-video-caltrain-releases-video-of-man-almost/vCTZdY/ | ||||
|             if 'playerKey' in url: | ||||
|                 return [url] | ||||
|  | ||||
|         matches = re.findall( | ||||
|             r'''(?sx)<object | ||||
| @@ -183,7 +203,7 @@ class BrightcoveIE(InfoExtractor): | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         self.report_extraction(video_id) | ||||
|         info = self._search_regex(r'var experienceJSON = ({.*?});', webpage, 'json') | ||||
|         info = self._search_regex(r'var experienceJSON = ({.*});', webpage, 'json') | ||||
|         info = json.loads(info)['data'] | ||||
|         video_info = info['programmedContent']['videoPlayer']['mediaDTO'] | ||||
|         video_info['_youtubedl_adServerURL'] = info.get('adServerURL') | ||||
| @@ -215,12 +235,26 @@ class BrightcoveIE(InfoExtractor): | ||||
|  | ||||
|         renditions = video_info.get('renditions') | ||||
|         if renditions: | ||||
|             renditions = sorted(renditions, key=lambda r: r['size']) | ||||
|             info['formats'] = [{ | ||||
|                 'url': rend['defaultURL'], | ||||
|                 'height': rend.get('frameHeight'), | ||||
|                 'width': rend.get('frameWidth'), | ||||
|             } for rend in renditions] | ||||
|             formats = [] | ||||
|             for rend in renditions: | ||||
|                 url = rend['defaultURL'] | ||||
|                 if rend['remote']: | ||||
|                     # This type of renditions are served through akamaihd.net, | ||||
|                     # but they don't use f4m manifests | ||||
|                     url = url.replace('control/', '') + '?&v=3.3.0&fp=13&r=FEEFJ&g=RTSJIMBMPFPB' | ||||
|                     ext = 'flv' | ||||
|                 else: | ||||
|                     ext = determine_ext(url) | ||||
|                 size = rend.get('size') | ||||
|                 formats.append({ | ||||
|                     'url': url, | ||||
|                     'ext': ext, | ||||
|                     'height': rend.get('frameHeight'), | ||||
|                     'width': rend.get('frameWidth'), | ||||
|                     'filesize': size if size != 0 else None, | ||||
|                 }) | ||||
|             self._sort_formats(formats) | ||||
|             info['formats'] = formats | ||||
|         elif video_info.get('FLVFullLengthURL') is not None: | ||||
|             info.update({ | ||||
|                 'url': video_info['FLVFullLengthURL'], | ||||
|   | ||||
							
								
								
									
										48
									
								
								youtube_dl/extractor/byutv.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								youtube_dl/extractor/byutv.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import json | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ExtractorError | ||||
|  | ||||
|  | ||||
| class BYUtvIE(InfoExtractor): | ||||
|     _VALID_URL = r'^https?://(?:www\.)?byutv.org/watch/[0-9a-f-]+/(?P<video_id>[^/?#]+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.byutv.org/watch/44e80f7b-e3ba-43ba-8c51-b1fd96c94a79/granite-flats-talking', | ||||
|         'info_dict': { | ||||
|             'id': 'granite-flats-talking', | ||||
|             'ext': 'mp4', | ||||
|             'description': 'md5:4e9a7ce60f209a33eca0ac65b4918e1c', | ||||
|             'title': 'Talking', | ||||
|             'thumbnail': 're:^https?://.*promo.*' | ||||
|         }, | ||||
|         'params': { | ||||
|             'skip_download': True, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('video_id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         episode_code = self._search_regex( | ||||
|             r'(?s)episode:(.*?\}),\s*\n', webpage, 'episode information') | ||||
|         episode_json = re.sub( | ||||
|             r'(\n\s+)([a-zA-Z]+):\s+\'(.*?)\'', r'\1"\2": "\3"', episode_code) | ||||
|         ep = json.loads(episode_json) | ||||
|  | ||||
|         if ep['providerType'] == 'Ooyala': | ||||
|             return { | ||||
|                 '_type': 'url_transparent', | ||||
|                 'ie_key': 'Ooyala', | ||||
|                 'url': 'ooyala:%s' % ep['providerId'], | ||||
|                 'id': video_id, | ||||
|                 'title': ep['title'], | ||||
|                 'description': ep.get('description'), | ||||
|                 'thumbnail': ep.get('imageThumbnail'), | ||||
|             } | ||||
|         else: | ||||
|             raise ExtractorError('Unsupported provider %s' % ep['provider']) | ||||
| @@ -2,39 +2,46 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import json | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class C56IE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://((www|player)\.)?56\.com/(.+?/)?(v_|(play_album.+-))(?P<textid>.+?)\.(html|swf)' | ||||
|     _VALID_URL = r'https?://(?:(?:www|player)\.)?56\.com/(?:.+?/)?(?:v_|(?:play_album.+-))(?P<textid>.+?)\.(?:html|swf)' | ||||
|     IE_NAME = '56.com' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.56.com/u39/v_OTM0NDA3MTY.html', | ||||
|         'file': '93440716.flv', | ||||
|         'md5': 'e59995ac63d0457783ea05f93f12a866', | ||||
|         'info_dict': { | ||||
|             'id': '93440716', | ||||
|             'ext': 'flv', | ||||
|             'title': '网事知多少 第32期:车怒', | ||||
|             'duration': 283.813, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url, flags=re.VERBOSE) | ||||
|         text_id = mobj.group('textid') | ||||
|         info_page = self._download_webpage('http://vxml.56.com/json/%s/' % text_id, | ||||
|                                            text_id, 'Downloading video info') | ||||
|         info = json.loads(info_page)['info'] | ||||
|         formats = [{ | ||||
|             'format_id': f['type'], | ||||
|             'filesize': int(f['filesize']), | ||||
|             'url': f['url'] | ||||
|         } for f in info['rfiles']] | ||||
|  | ||||
|         page = self._download_json( | ||||
|             'http://vxml.56.com/json/%s/' % text_id, text_id, 'Downloading video info') | ||||
|  | ||||
|         info = page['info'] | ||||
|  | ||||
|         formats = [ | ||||
|             { | ||||
|                 'format_id': f['type'], | ||||
|                 'filesize': int(f['filesize']), | ||||
|                 'url': f['url'] | ||||
|             } for f in info['rfiles'] | ||||
|         ] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': info['vid'], | ||||
|             'title': info['Subject'], | ||||
|             'duration': int(info['duration']) / 1000.0, | ||||
|             'formats': formats, | ||||
|             'thumbnail': info.get('bimg') or info.get('img'), | ||||
|         } | ||||
|   | ||||
| @@ -1,53 +1,72 @@ | ||||
| # encoding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import unified_strdate | ||||
| from ..utils import ( | ||||
|     unified_strdate, | ||||
|     url_basename, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class CanalplusIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(www\.canalplus\.fr/.*?/(?P<path>.*)|player\.canalplus\.fr/#/(?P<id>\d+))' | ||||
|     _VALID_URL = r'https?://(?:www\.canalplus\.fr/.*?/(?P<path>.*)|player\.canalplus\.fr/#/(?P<id>[0-9]+))' | ||||
|     _VIDEO_INFO_TEMPLATE = 'http://service.canal-plus.com/video/rest/getVideosLiees/cplus/%s' | ||||
|     IE_NAME = u'canalplus.fr' | ||||
|     IE_NAME = 'canalplus.fr' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.canalplus.fr/c-infos-documentaires/pid1830-c-zapping.html?vid=922470', | ||||
|         u'file': u'922470.flv', | ||||
|         u'info_dict': { | ||||
|             u'title': u'Zapping - 26/08/13', | ||||
|             u'description': u'Le meilleur de toutes les chaînes, tous les jours.\nEmission du 26 août 2013', | ||||
|             u'upload_date': u'20130826', | ||||
|         }, | ||||
|         u'params': { | ||||
|             u'skip_download': True, | ||||
|         'url': 'http://www.canalplus.fr/c-infos-documentaires/pid1830-c-zapping.html?vid=922470', | ||||
|         'md5': '3db39fb48b9685438ecf33a1078023e4', | ||||
|         'info_dict': { | ||||
|             'id': '922470', | ||||
|             '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', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.groupdict().get('id') | ||||
|  | ||||
|         # Beware, some subclasses do not define an id group | ||||
|         display_id = url_basename(mobj.group('path')) | ||||
|  | ||||
|         if video_id is None: | ||||
|             webpage = self._download_webpage(url, mobj.group('path')) | ||||
|             video_id = self._search_regex(r'videoId = "(\d+)";', webpage, u'video id') | ||||
|             webpage = self._download_webpage(url, display_id) | ||||
|             video_id = self._search_regex(r'<canal:player videoId="(\d+)"', webpage, 'video id') | ||||
|  | ||||
|         info_url = self._VIDEO_INFO_TEMPLATE % video_id | ||||
|         doc = self._download_xml(info_url,video_id,  | ||||
|                                            u'Downloading video info') | ||||
|         doc = self._download_xml(info_url, video_id, 'Downloading video XML') | ||||
|  | ||||
|         self.report_extraction(video_id) | ||||
|         video_info = [video for video in doc if video.find('ID').text == video_id][0] | ||||
|         infos = video_info.find('INFOS') | ||||
|         media = video_info.find('MEDIA') | ||||
|         formats = [media.find('VIDEOS/%s' % format) | ||||
|             for format in ['BAS_DEBIT', 'HAUT_DEBIT', 'HD']] | ||||
|         video_url = [format.text for format in formats if format is not None][-1] | ||||
|         infos = video_info.find('INFOS') | ||||
|  | ||||
|         return {'id': video_id, | ||||
|                 'title': u'%s - %s' % (infos.find('TITRAGE/TITRE').text, | ||||
|                                        infos.find('TITRAGE/SOUS_TITRE').text), | ||||
|                 'url': video_url, | ||||
|                 'ext': 'flv', | ||||
|                 'upload_date': unified_strdate(infos.find('PUBLICATION/DATE').text), | ||||
|                 'thumbnail': media.find('IMAGES/GRAND').text, | ||||
|                 'description': infos.find('DESCRIPTION').text, | ||||
|                 'view_count': int(infos.find('NB_VUES').text), | ||||
|                 } | ||||
|         preferences = ['MOBILE', 'BAS_DEBIT', 'HAUT_DEBIT', 'HD', 'HLS', 'HDS'] | ||||
|  | ||||
|         formats = [ | ||||
|             { | ||||
|                 'url': fmt.text + '?hdcore=2.11.3' if fmt.tag == 'HDS' else fmt.text, | ||||
|                 'format_id': fmt.tag, | ||||
|                 'ext': 'mp4' if fmt.tag == 'HLS' else 'flv', | ||||
|                 'preference': preferences.index(fmt.tag) if fmt.tag in preferences else -1, | ||||
|             } for fmt in media.find('VIDEOS') if fmt.text | ||||
|         ] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'display_id': display_id, | ||||
|             'title': '%s - %s' % (infos.find('TITRAGE/TITRE').text, | ||||
|                                   infos.find('TITRAGE/SOUS_TITRE').text), | ||||
|             'upload_date': unified_strdate(infos.find('PUBLICATION/DATE').text), | ||||
|             'thumbnail': media.find('IMAGES/GRAND').text, | ||||
|             'description': infos.find('DESCRIPTION').text, | ||||
|             'view_count': int(infos.find('NB_VUES').text), | ||||
|             'like_count': int(infos.find('NB_LIKES').text), | ||||
|             'comment_count': int(infos.find('NB_COMMENTS').text), | ||||
|             'formats': formats, | ||||
|         } | ||||
							
								
								
									
										87
									
								
								youtube_dl/extractor/cbsnews.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								youtube_dl/extractor/cbsnews.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| # encoding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import json | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class CBSNewsIE(InfoExtractor): | ||||
|     IE_DESC = 'CBS News' | ||||
|     _VALID_URL = r'http://(?:www\.)?cbsnews\.com/(?:[^/]+/)+(?P<id>[\da-z_-]+)' | ||||
|  | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             'url': 'http://www.cbsnews.com/news/tesla-and-spacex-elon-musks-industrial-empire/', | ||||
|             'info_dict': { | ||||
|                 'id': 'tesla-and-spacex-elon-musks-industrial-empire', | ||||
|                 'ext': 'flv', | ||||
|                 'title': 'Tesla and SpaceX: Elon Musk\'s industrial empire', | ||||
|                 'thumbnail': 'http://beta.img.cbsnews.com/i/2014/03/30/60147937-2f53-4565-ad64-1bdd6eb64679/60-0330-pelley-640x360.jpg', | ||||
|                 'duration': 791, | ||||
|             }, | ||||
|             'params': { | ||||
|                 # rtmp download | ||||
|                 'skip_download': True, | ||||
|             }, | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://www.cbsnews.com/videos/fort-hood-shooting-army-downplays-mental-illness-as-cause-of-attack/', | ||||
|             'info_dict': { | ||||
|                 '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', | ||||
|                 'duration': 205, | ||||
|             }, | ||||
|             'params': { | ||||
|                 # rtmp download | ||||
|                 'skip_download': 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) | ||||
|  | ||||
|         video_info = json.loads(self._html_search_regex( | ||||
|             r'(?:<ul class="media-list items" id="media-related-items"><li data-video-info|<div id="cbsNewsVideoPlayer" data-video-player-options)=\'({.+?})\'', | ||||
|             webpage, 'video JSON info')) | ||||
|  | ||||
|         item = video_info['item'] if 'item' in video_info else video_info | ||||
|         title = item.get('articleTitle') or item.get('hed') | ||||
|         duration = item.get('duration') | ||||
|         thumbnail = item.get('mediaImage') or item.get('thumbnail') | ||||
|  | ||||
|         formats = [] | ||||
|         for format_id in ['RtmpMobileLow', 'RtmpMobileHigh', 'Hls', 'RtmpDesktop']: | ||||
|             uri = item.get('media' + format_id + 'URI') | ||||
|             if not uri: | ||||
|                 continue | ||||
|             fmt = { | ||||
|                 'url': uri, | ||||
|                 'format_id': format_id, | ||||
|             } | ||||
|             if uri.startswith('rtmp'): | ||||
|                 fmt.update({ | ||||
|                     'app': 'ondemand?auth=cbs', | ||||
|                     'play_path': 'mp4:' + uri.split('<break>')[-1], | ||||
|                     'player_url': 'http://www.cbsnews.com/[[IMPORT]]/vidtech.cbsinteractive.com/player/3_3_0/CBSI_PLAYER_HD.swf', | ||||
|                     'page_url': 'http://www.cbsnews.com', | ||||
|                     'ext': 'flv', | ||||
|                 }) | ||||
|             elif uri.endswith('.m3u8'): | ||||
|                 fmt['ext'] = 'mp4' | ||||
|             formats.append(fmt) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'duration': duration, | ||||
|             'formats': formats, | ||||
|         } | ||||
| @@ -1,21 +1,24 @@ | ||||
| # encoding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     int_or_none, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class CinemassacreIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://(?:www\.)?cinemassacre\.com/(?P<date_Y>[0-9]{4})/(?P<date_m>[0-9]{2})/(?P<date_d>[0-9]{2})/.+?' | ||||
|     _VALID_URL = r'http://(?: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/', | ||||
|             'file': '19911.mp4', | ||||
|             'md5': 'fde81fbafaee331785f58cd6c0d46190', | ||||
|             'info_dict': { | ||||
|                 'id': '19911', | ||||
|                 'ext': 'mp4', | ||||
|                 'upload_date': '20121110', | ||||
|                 'title': '“Angry Video Game Nerd: The Movie” – Trailer', | ||||
|                 'description': 'md5:fb87405fcb42a331742a0dce2708560b', | ||||
| @@ -23,9 +26,10 @@ class CinemassacreIE(InfoExtractor): | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://cinemassacre.com/2013/10/02/the-mummys-hand-1940', | ||||
|             'file': '521be8ef82b16.mp4', | ||||
|             'md5': 'd72f10cd39eac4215048f62ab477a511', | ||||
|             'info_dict': { | ||||
|                 'id': '521be8ef82b16', | ||||
|                 'ext': 'mp4', | ||||
|                 'upload_date': '20131002', | ||||
|                 'title': 'The Mummy’s Hand (1940)', | ||||
|             }, | ||||
| @@ -34,8 +38,9 @@ class CinemassacreIE(InfoExtractor): | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         display_id = mobj.group('display_id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, None)  # Don't know video id yet | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|         video_date = mobj.group('date_Y') + mobj.group('date_m') + mobj.group('date_d') | ||||
|         mobj = re.search(r'src="(?P<embed_url>http://player\.screenwavemedia\.com/play/[a-zA-Z]+\.php\?id=(?:Cinemassacre-)?(?P<video_id>.+?))"', webpage) | ||||
|         if not mobj: | ||||
| @@ -43,33 +48,47 @@ class CinemassacreIE(InfoExtractor): | ||||
|         playerdata_url = mobj.group('embed_url') | ||||
|         video_id = mobj.group('video_id') | ||||
|  | ||||
|         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>', | ||||
|         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) | ||||
|         if len(video_description) == 0: | ||||
|             video_description = None | ||||
|  | ||||
|         playerdata = self._download_webpage(playerdata_url, video_id) | ||||
|         playerdata = self._download_webpage(playerdata_url, video_id, 'Downloading player webpage') | ||||
|         video_thumbnail = self._search_regex( | ||||
|             r'image: \'(?P<thumbnail>[^\']+)\'', playerdata, 'thumbnail', fatal=False) | ||||
|         sd_url = self._search_regex(r'file: \'([^\']+)\', label: \'SD\'', playerdata, 'sd_file') | ||||
|         videolist_url = self._search_regex(r'file: \'([^\']+\.smil)\'}', playerdata, 'videolist_url') | ||||
|  | ||||
|         sd_url = self._html_search_regex(r'file: \'(?P<sd_file>[^\']+)\', label: \'SD\'', playerdata, 'sd_file') | ||||
|         hd_url = self._html_search_regex(r'file: \'(?P<hd_file>[^\']+)\', label: \'HD\'', playerdata, 'hd_file') | ||||
|         video_thumbnail = self._html_search_regex(r'image: \'(?P<thumbnail>[^\']+)\'', playerdata, 'thumbnail', fatal=False) | ||||
|         videolist = self._download_xml(videolist_url, video_id, 'Downloading videolist XML') | ||||
|  | ||||
|         formats = [ | ||||
|             { | ||||
|                 'url': sd_url, | ||||
|                 'ext': 'mp4', | ||||
|                 'format': 'sd', | ||||
|                 'format_id': 'sd', | ||||
|             }, | ||||
|             { | ||||
|                 'url': hd_url, | ||||
|                 'ext': 'mp4', | ||||
|                 'format': 'hd', | ||||
|                 'format_id': 'hd', | ||||
|             }, | ||||
|         ] | ||||
|         formats = [] | ||||
|         baseurl = sd_url[:sd_url.rfind('/')+1] | ||||
|         for video in videolist.findall('.//video'): | ||||
|             src = video.get('src') | ||||
|             if not src: | ||||
|                 continue | ||||
|             file_ = src.partition(':')[-1] | ||||
|             width = int_or_none(video.get('width')) | ||||
|             height = int_or_none(video.get('height')) | ||||
|             bitrate = int_or_none(video.get('system-bitrate')) | ||||
|             format = { | ||||
|                 'url': baseurl + file_, | ||||
|                 'format_id': src.rpartition('.')[0].rpartition('_')[-1], | ||||
|             } | ||||
|             if width or height: | ||||
|                 format.update({ | ||||
|                     'tbr': bitrate // 1000 if bitrate else None, | ||||
|                     'width': width, | ||||
|                     'height': height, | ||||
|                 }) | ||||
|             else: | ||||
|                 format.update({ | ||||
|                     'abr': bitrate // 1000 if bitrate else None, | ||||
|                     'vcodec': 'none', | ||||
|                 }) | ||||
|             formats.append(format) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|   | ||||
| @@ -1,22 +1,28 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import time | ||||
| import xml.etree.ElementTree | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ExtractorError | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     parse_duration, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class ClipfishIE(InfoExtractor): | ||||
|     IE_NAME = u'clipfish' | ||||
|     IE_NAME = 'clipfish' | ||||
|  | ||||
|     _VALID_URL = r'^https?://(?:www\.)?clipfish\.de/.*?/video/(?P<id>[0-9]+)/' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.clipfish.de/special/game-trailer/video/3966754/fifa-14-e3-2013-trailer/', | ||||
|         u'file': u'3966754.mp4', | ||||
|         u'md5': u'2521cd644e862936cf2e698206e47385', | ||||
|         u'info_dict': { | ||||
|             u'title': u'FIFA 14 - E3 2013 Trailer', | ||||
|             u'duration': 82, | ||||
|         'url': 'http://www.clipfish.de/special/game-trailer/video/3966754/fifa-14-e3-2013-trailer/', | ||||
|         'md5': '2521cd644e862936cf2e698206e47385', | ||||
|         'info_dict': { | ||||
|             'id': '3966754', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'FIFA 14 - E3 2013 Trailer', | ||||
|             'duration': 82, | ||||
|         }, | ||||
|         u'skip': 'Blocked in the US' | ||||
|     } | ||||
| @@ -33,21 +39,10 @@ class ClipfishIE(InfoExtractor): | ||||
|         video_url = doc.find('filename').text | ||||
|         if video_url is None: | ||||
|             xml_bytes = xml.etree.ElementTree.tostring(doc) | ||||
|             raise ExtractorError(u'Cannot find video URL in document %r' % | ||||
|             raise ExtractorError('Cannot find video URL in document %r' % | ||||
|                                  xml_bytes) | ||||
|         thumbnail = doc.find('imageurl').text | ||||
|         duration_str = doc.find('duration').text | ||||
|         m = re.match( | ||||
|             r'^(?P<hours>[0-9]+):(?P<minutes>[0-9]{2}):(?P<seconds>[0-9]{2}):(?P<ms>[0-9]*)$', | ||||
|             duration_str) | ||||
|         if m: | ||||
|             duration = ( | ||||
|                 (int(m.group('hours')) * 60 * 60) + | ||||
|                 (int(m.group('minutes')) * 60) + | ||||
|                 (int(m.group('seconds'))) | ||||
|             ) | ||||
|         else: | ||||
|             duration = None | ||||
|         duration = parse_duration(doc.find('duration').text) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| @@ -11,13 +13,14 @@ class ClipsyndicateIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://www\.clipsyndicate\.com/video/play(list/\d+)?/(?P<id>\d+)' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.clipsyndicate.com/video/play/4629301/brick_briscoe', | ||||
|         u'md5': u'4d7d549451bad625e0ff3d7bd56d776c', | ||||
|         u'info_dict': { | ||||
|             u'id': u'4629301', | ||||
|             u'ext': u'mp4', | ||||
|             u'title': u'Brick Briscoe', | ||||
|             u'duration': 612, | ||||
|         'url': 'http://www.clipsyndicate.com/video/play/4629301/brick_briscoe', | ||||
|         'md5': '4d7d549451bad625e0ff3d7bd56d776c', | ||||
|         'info_dict': { | ||||
|             'id': '4629301', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Brick Briscoe', | ||||
|             'duration': 612, | ||||
|             'thumbnail': 're:^https?://.+\.jpg', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
| @@ -26,13 +29,13 @@ class ClipsyndicateIE(InfoExtractor): | ||||
|         video_id = mobj.group('id') | ||||
|         js_player = self._download_webpage( | ||||
|             'http://eplayer.clipsyndicate.com/embed/player.js?va_id=%s' % video_id, | ||||
|             video_id, u'Downlaoding player') | ||||
|             video_id, 'Downlaoding player') | ||||
|         # it includes a required token | ||||
|         flvars = self._search_regex(r'flvars: "(.*?)"', js_player, u'flvars') | ||||
|         flvars = self._search_regex(r'flvars: "(.*?)"', js_player, 'flvars') | ||||
|  | ||||
|         pdoc = self._download_xml( | ||||
|             'http://eplayer.clipsyndicate.com/osmf/playlist?%s' % flvars, | ||||
|             video_id, u'Downloading video info', | ||||
|             video_id, 'Downloading video info', | ||||
|             transform_source=fix_xml_ampersands) | ||||
|  | ||||
|         track_doc = pdoc.find('trackList/track') | ||||
|   | ||||
							
								
								
									
										58
									
								
								youtube_dl/extractor/clubic.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								youtube_dl/extractor/clubic.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import json | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     clean_html, | ||||
|     qualities, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class ClubicIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://(?:www\.)?clubic\.com/video/[^/]+/video.*-(?P<id>[0-9]+)\.html' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.clubic.com/video/clubic-week/video-clubic-week-2-0-le-fbi-se-lance-dans-la-photo-d-identite-448474.html', | ||||
|         'md5': '1592b694ba586036efac1776b0b43cd3', | ||||
|         'info_dict': { | ||||
|             'id': '448474', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Clubic Week 2.0 : le FBI se lance dans la photo d\u0092identité', | ||||
|             'description': 're:Gueule de bois chez Nokia. Le constructeur a indiqué cette.*', | ||||
|             'thumbnail': 're:^http://img\.clubic\.com/.*\.jpg$', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         player_url = 'http://player.m6web.fr/v1/player/clubic/%s.html' % video_id | ||||
|         player_page = self._download_webpage(player_url, video_id) | ||||
|  | ||||
|         config_json = self._search_regex( | ||||
|             r'(?m)M6\.Player\.config\s*=\s*(\{.+?\});$', player_page, | ||||
|             'configuration') | ||||
|         config = json.loads(config_json) | ||||
|  | ||||
|         video_info = config['videoInfo'] | ||||
|         sources = config['sources'] | ||||
|         quality_order = qualities(['sd', 'hq']) | ||||
|  | ||||
|         formats = [{ | ||||
|             'format_id': src['streamQuality'], | ||||
|             'url': src['src'], | ||||
|             'quality': quality_order(src['streamQuality']), | ||||
|         } for src in sources] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': video_info['title'], | ||||
|             'formats': formats, | ||||
|             'description': clean_html(video_info.get('description')), | ||||
|             'thumbnail': config.get('poster'), | ||||
|         } | ||||
| @@ -1,19 +1,19 @@ | ||||
| from __future__ import unicode_literals | ||||
| from .mtv import MTVIE | ||||
|  | ||||
|  | ||||
| class CMTIE(MTVIE): | ||||
|     IE_NAME = u'cmt.com' | ||||
|     IE_NAME = 'cmt.com' | ||||
|     _VALID_URL = r'https?://www\.cmt\.com/videos/.+?/(?P<videoid>[^/]+)\.jhtml' | ||||
|     _FEED_URL = 'http://www.cmt.com/sitewide/apps/player/embed/rss/' | ||||
|  | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             u'url': u'http://www.cmt.com/videos/garth-brooks/989124/the-call-featuring-trisha-yearwood.jhtml#artist=30061', | ||||
|             u'md5': u'e6b7ef3c4c45bbfae88061799bbba6c2', | ||||
|             u'info_dict': { | ||||
|                 u'id': u'989124', | ||||
|                 u'ext': u'mp4', | ||||
|                 u'title': u'Garth Brooks - "The Call (featuring Trisha Yearwood)"', | ||||
|                 u'description': u'Blame It All On My Roots', | ||||
|             }, | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.cmt.com/videos/garth-brooks/989124/the-call-featuring-trisha-yearwood.jhtml#artist=30061', | ||||
|         'md5': 'e6b7ef3c4c45bbfae88061799bbba6c2', | ||||
|         'info_dict': { | ||||
|             'id': '989124', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Garth Brooks - "The Call (featuring Trisha Yearwood)"', | ||||
|             'description': 'Blame It All On My Roots', | ||||
|         }, | ||||
|     ] | ||||
|     }] | ||||
|   | ||||
							
								
								
									
										75
									
								
								youtube_dl/extractor/cnet.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								youtube_dl/extractor/cnet.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import json | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     int_or_none, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class CNETIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?cnet\.com/videos/(?P<id>[^/]+)/' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.cnet.com/videos/hands-on-with-microsofts-windows-8-1-update/', | ||||
|         'md5': '041233212a0d06b179c87cbcca1577b8', | ||||
|         'info_dict': { | ||||
|             'id': '56f4ea68-bd21-4852-b08c-4de5b8354c60', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Hands-on with Microsoft Windows 8.1 Update', | ||||
|             'description': 'The new update to the Windows 8 OS brings improved performance for mouse and keyboard users.', | ||||
|             'thumbnail': 're:^http://.*/flmswindows8.jpg$', | ||||
|             'uploader_id': 'sarah.mitroff@cbsinteractive.com', | ||||
|             'uploader': 'Sarah Mitroff', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         display_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|         data_json = self._html_search_regex( | ||||
|             r"<div class=\"cnetVideoPlayer\"\s+.*?data-cnet-video-options='([^']+)'", | ||||
|             webpage, 'data json') | ||||
|         data = json.loads(data_json) | ||||
|         vdata = data['video'] | ||||
|         if not vdata: | ||||
|             vdata = data['videos'][0] | ||||
|         if not vdata: | ||||
|             raise ExtractorError('Cannot find video data') | ||||
|  | ||||
|         video_id = vdata['id'] | ||||
|         title = vdata['headline'] | ||||
|         description = vdata.get('dek') | ||||
|         thumbnail = vdata.get('image', {}).get('path') | ||||
|         author = vdata.get('author') | ||||
|         if author: | ||||
|             uploader = '%s %s' % (author['firstName'], author['lastName']) | ||||
|             uploader_id = author.get('email') | ||||
|         else: | ||||
|             uploader = None | ||||
|             uploader_id = None | ||||
|  | ||||
|         formats = [{ | ||||
|             'format_id': '%s-%s-%s' % ( | ||||
|                 f['type'], f['format'], | ||||
|                 int_or_none(f.get('bitrate'), 1000, default='')), | ||||
|             'url': f['uri'], | ||||
|             'tbr': int_or_none(f.get('bitrate'), 1000), | ||||
|         } for f in vdata['files']['data']] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'display_id': display_id, | ||||
|             'title': title, | ||||
|             'formats': formats, | ||||
|             'description': description, | ||||
|             'uploader': uploader, | ||||
|             'uploader_id': uploader_id, | ||||
|             'thumbnail': thumbnail, | ||||
|         } | ||||
| @@ -79,8 +79,11 @@ class CNNIE(InfoExtractor): | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         thumbnails = sorted([((int(t.attrib['height']),int(t.attrib['width'])), t.text) for t in info.findall('images/image')]) | ||||
|         thumbs_dict = [{'resolution': res, 'url': t_url} for (res, t_url) in thumbnails] | ||||
|         thumbnails = [{ | ||||
|             'height': int(t.attrib['height']), | ||||
|             'width': int(t.attrib['width']), | ||||
|             'url': t.text, | ||||
|         } for t in info.findall('images/image')] | ||||
|  | ||||
|         metas_el = info.find('metas') | ||||
|         upload_date = ( | ||||
| @@ -93,8 +96,7 @@ class CNNIE(InfoExtractor): | ||||
|             'id': info.attrib['id'], | ||||
|             'title': info.find('headline').text, | ||||
|             'formats': formats, | ||||
|             'thumbnail': thumbnails[-1][1], | ||||
|             'thumbnails': thumbs_dict, | ||||
|             'thumbnails': thumbnails, | ||||
|             'description': info.find('description').text, | ||||
|             'duration': duration, | ||||
|             'upload_date': upload_date, | ||||
|   | ||||
| @@ -7,8 +7,8 @@ from .mtv import MTVServicesInfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_str, | ||||
|     compat_urllib_parse, | ||||
|  | ||||
|     ExtractorError, | ||||
|     float_or_none, | ||||
|     unified_strdate, | ||||
| ) | ||||
|  | ||||
| @@ -21,7 +21,7 @@ class ComedyCentralIE(MTVServicesInfoExtractor): | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.comedycentral.com/video-clips/kllhuv/stand-up-greg-fitzsimmons--uncensored---too-good-of-a-mother', | ||||
|         'md5': '4167875aae411f903b751a21f357f1ee', | ||||
|         'md5': 'c4f48e9eda1b16dd10add0744344b6d8', | ||||
|         'info_dict': { | ||||
|             'id': 'cef0cbb3-e776-4bc9-b62e-8016deccb354', | ||||
|             'ext': 'mp4', | ||||
| @@ -32,31 +32,34 @@ class ComedyCentralIE(MTVServicesInfoExtractor): | ||||
|  | ||||
|  | ||||
| class ComedyCentralShowsIE(InfoExtractor): | ||||
|     IE_DESC = 'The Daily Show / Colbert Report' | ||||
|     IE_DESC = 'The Daily Show / The Colbert Report' | ||||
|     # urls can be abbreviations like :thedailyshow or :colbert | ||||
|     # urls for episodes like: | ||||
|     # or urls for clips like: http://www.thedailyshow.com/watch/mon-december-10-2012/any-given-gun-day | ||||
|     #                     or: http://www.colbertnation.com/the-colbert-report-videos/421667/november-29-2012/moon-shattering-news | ||||
|     #                     or: http://www.colbertnation.com/the-colbert-report-collections/422008/festival-of-lights/79524 | ||||
|     _VALID_URL = r"""^(:(?P<shortname>tds|thedailyshow|cr|colbert|colbertnation|colbertreport) | ||||
|                       |(https?://)?(www\.)? | ||||
|                           (?P<showname>thedailyshow|colbertnation)\.com/ | ||||
|                          (full-episodes/(?P<episode>.*)| | ||||
|     _VALID_URL = r'''(?x)^(:(?P<shortname>tds|thedailyshow|cr|colbert|colbertnation|colbertreport) | ||||
|                       |https?://(:www\.)? | ||||
|                           (?P<showname>thedailyshow|thecolbertreport)\.(?:cc\.)?com/ | ||||
|                          ((?:full-)?episodes/(?:[0-9a-z]{6}/)?(?P<episode>.*)| | ||||
|                           (?P<clip> | ||||
|                               (the-colbert-report-(videos|collections)/(?P<clipID>[0-9]+)/[^/]*/(?P<cntitle>.*?)) | ||||
|                               |(watch/(?P<date>[^/]*)/(?P<tdstitle>.*)))| | ||||
|                               (?:(?:guests/[^/]+|videos|video-playlists|special-editions)/[^/]+/(?P<videotitle>[^/?#]+)) | ||||
|                               |(the-colbert-report-(videos|collections)/(?P<clipID>[0-9]+)/[^/]*/(?P<cntitle>.*?)) | ||||
|                               |(watch/(?P<date>[^/]*)/(?P<tdstitle>.*)) | ||||
|                           )| | ||||
|                           (?P<interview> | ||||
|                               extended-interviews/(?P<interID>[0-9]+)/playlist_tds_extended_(?P<interview_title>.*?)/.*?))) | ||||
|                      $""" | ||||
|                               extended-interviews/(?P<interID>[0-9a-z]+)/(?:playlist_tds_extended_)?(?P<interview_title>.*?)(/.*?)?))) | ||||
|                      (?:[?#].*|$)''' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.thedailyshow.com/watch/thu-december-13-2012/kristen-stewart', | ||||
|         'file': '422212.mp4', | ||||
|         'url': 'http://thedailyshow.cc.com/watch/thu-december-13-2012/kristen-stewart', | ||||
|         'md5': '4e2f5cb088a83cd8cdb7756132f9739d', | ||||
|         'info_dict': { | ||||
|             "upload_date": "20121214", | ||||
|             "description": "Kristen Stewart", | ||||
|             "uploader": "thedailyshow", | ||||
|             "title": "thedailyshow-kristen-stewart part 1" | ||||
|             'id': 'ab9ab3e7-5a98-4dbe-8b21-551dc0523d55', | ||||
|             'ext': 'mp4', | ||||
|             'upload_date': '20121213', | ||||
|             'description': 'Kristen Stewart learns to let loose in "On the Road."', | ||||
|             'uploader': 'thedailyshow', | ||||
|             'title': 'thedailyshow kristen-stewart part 1', | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -79,11 +82,6 @@ class ComedyCentralShowsIE(InfoExtractor): | ||||
|         '400': (384, 216), | ||||
|     } | ||||
|  | ||||
|     @classmethod | ||||
|     def suitable(cls, url): | ||||
|         """Receives a URL and returns True if suitable for this IE.""" | ||||
|         return re.match(cls._VALID_URL, url, re.VERBOSE) is not None | ||||
|  | ||||
|     @staticmethod | ||||
|     def _transform_rtmp_url(rtmp_video_url): | ||||
|         m = re.match(r'^rtmpe?://.*?/(?P<finalid>gsp\.comedystor/.*)$', rtmp_video_url) | ||||
| @@ -99,14 +97,16 @@ class ComedyCentralShowsIE(InfoExtractor): | ||||
|  | ||||
|         if mobj.group('shortname'): | ||||
|             if mobj.group('shortname') in ('tds', 'thedailyshow'): | ||||
|                 url = 'http://www.thedailyshow.com/full-episodes/' | ||||
|                 url = 'http://thedailyshow.cc.com/full-episodes/' | ||||
|             else: | ||||
|                 url = 'http://www.colbertnation.com/full-episodes/' | ||||
|                 url = 'http://thecolbertreport.cc.com/full-episodes/' | ||||
|             mobj = re.match(self._VALID_URL, url, re.VERBOSE) | ||||
|             assert mobj is not None | ||||
|  | ||||
|         if mobj.group('clip'): | ||||
|             if mobj.group('showname') == 'thedailyshow': | ||||
|             if mobj.group('videotitle'): | ||||
|                 epTitle = mobj.group('videotitle') | ||||
|             elif mobj.group('showname') == 'thedailyshow': | ||||
|                 epTitle = mobj.group('tdstitle') | ||||
|             else: | ||||
|                 epTitle = mobj.group('cntitle') | ||||
| @@ -120,9 +120,9 @@ class ComedyCentralShowsIE(InfoExtractor): | ||||
|                 epTitle = mobj.group('showname') | ||||
|             else: | ||||
|                 epTitle = mobj.group('episode') | ||||
|         show_name = mobj.group('showname') | ||||
|  | ||||
|         self.report_extraction(epTitle) | ||||
|         webpage,htmlHandle = self._download_webpage_handle(url, epTitle) | ||||
|         webpage, htmlHandle = self._download_webpage_handle(url, epTitle) | ||||
|         if dlNewest: | ||||
|             url = htmlHandle.geturl() | ||||
|             mobj = re.match(self._VALID_URL, url, re.VERBOSE) | ||||
| @@ -130,71 +130,86 @@ class ComedyCentralShowsIE(InfoExtractor): | ||||
|                 raise ExtractorError('Invalid redirected URL: ' + url) | ||||
|             if mobj.group('episode') == '': | ||||
|                 raise ExtractorError('Redirected URL is still not specific: ' + url) | ||||
|             epTitle = mobj.group('episode') | ||||
|             epTitle = (mobj.group('episode') or mobj.group('videotitle')).rpartition('/')[-1] | ||||
|  | ||||
|         mMovieParams = re.findall('(?:<param name="movie" value="|var url = ")(http://media.mtvnservices.com/([^"]*(?:episode|video).*?:.*?))"', webpage) | ||||
|  | ||||
|         if len(mMovieParams) == 0: | ||||
|             # The Colbert Report embeds the information in a without | ||||
|             # a URL prefix; so extract the alternate reference | ||||
|             # and then add the URL prefix manually. | ||||
|  | ||||
|             altMovieParams = re.findall('data-mgid="([^"]*(?:episode|video).*?:.*?)"', webpage) | ||||
|             altMovieParams = re.findall('data-mgid="([^"]*(?:episode|video|playlist).*?:.*?)"', webpage) | ||||
|             if len(altMovieParams) == 0: | ||||
|                 raise ExtractorError('unable to find Flash URL in webpage ' + url) | ||||
|             else: | ||||
|                 mMovieParams = [("http://media.mtvnservices.com/" + altMovieParams[0], altMovieParams[0])] | ||||
|  | ||||
|         uri = mMovieParams[0][1] | ||||
|         indexUrl = 'http://shadow.comedycentral.com/feeds/video_player/mrss/?' + compat_urllib_parse.urlencode({'uri': uri}) | ||||
|         idoc = self._download_xml(indexUrl, epTitle, | ||||
|                                           'Downloading show index', | ||||
|                                           'unable to download episode index') | ||||
|         # Correct cc.com in uri | ||||
|         uri = re.sub(r'(episode:[^.]+)(\.cc)?\.com', r'\1.cc.com', uri) | ||||
|  | ||||
|         results = [] | ||||
|         index_url = 'http://%s.cc.com/feeds/mrss?%s' % (show_name, compat_urllib_parse.urlencode({'uri': uri})) | ||||
|         idoc = self._download_xml( | ||||
|             index_url, epTitle, | ||||
|             'Downloading show index', 'Unable to download episode index') | ||||
|  | ||||
|         itemEls = idoc.findall('.//item') | ||||
|         for partNum,itemEl in enumerate(itemEls): | ||||
|             mediaId = itemEl.findall('./guid')[0].text | ||||
|             shortMediaId = mediaId.split(':')[-1] | ||||
|             showId = mediaId.split(':')[-2].replace('.com', '') | ||||
|             officialTitle = itemEl.findall('./title')[0].text | ||||
|             officialDate = unified_strdate(itemEl.findall('./pubDate')[0].text) | ||||
|         title = idoc.find('./channel/title').text | ||||
|         description = idoc.find('./channel/description').text | ||||
|  | ||||
|             configUrl = ('http://www.comedycentral.com/global/feeds/entertainment/media/mediaGenEntertainment.jhtml?' + | ||||
|                         compat_urllib_parse.urlencode({'uri': mediaId})) | ||||
|             cdoc = self._download_xml(configUrl, epTitle, | ||||
|                                                'Downloading configuration for %s' % shortMediaId) | ||||
|         entries = [] | ||||
|         item_els = idoc.findall('.//item') | ||||
|         for part_num, itemEl in enumerate(item_els): | ||||
|             upload_date = unified_strdate(itemEl.findall('./pubDate')[0].text) | ||||
|             thumbnail = itemEl.find('.//{http://search.yahoo.com/mrss/}thumbnail').attrib.get('url') | ||||
|  | ||||
|             content = itemEl.find('.//{http://search.yahoo.com/mrss/}content') | ||||
|             duration = float_or_none(content.attrib.get('duration')) | ||||
|             mediagen_url = content.attrib['url'] | ||||
|             guid = itemEl.find('./guid').text.rpartition(':')[-1] | ||||
|  | ||||
|             cdoc = self._download_xml( | ||||
|                 mediagen_url, epTitle, | ||||
|                 'Downloading configuration for segment %d / %d' % (part_num + 1, len(item_els))) | ||||
|  | ||||
|             turls = [] | ||||
|             for rendition in cdoc.findall('.//rendition'): | ||||
|                 finfo = (rendition.attrib['bitrate'], rendition.findall('./src')[0].text) | ||||
|                 turls.append(finfo) | ||||
|  | ||||
|             if len(turls) == 0: | ||||
|                 self._downloader.report_error('unable to download ' + mediaId + ': No videos found') | ||||
|                 continue | ||||
|  | ||||
|             formats = [] | ||||
|             for format, rtmp_video_url in turls: | ||||
|                 w, h = self._video_dimensions.get(format, (None, None)) | ||||
|                 formats.append({ | ||||
|                     'format_id': 'vhttp-%s' % format, | ||||
|                     'url': self._transform_rtmp_url(rtmp_video_url), | ||||
|                     'ext': self._video_extensions.get(format, 'mp4'), | ||||
|                     'format_id': format, | ||||
|                     'height': h, | ||||
|                     'width': w, | ||||
|                 }) | ||||
|                 formats.append({ | ||||
|                     'format_id': 'rtmp-%s' % format, | ||||
|                     'url': rtmp_video_url.replace('viacomccstrm', 'viacommtvstrm'), | ||||
|                     'ext': self._video_extensions.get(format, 'mp4'), | ||||
|                     'height': h, | ||||
|                     'width': w, | ||||
|                 }) | ||||
|                 self._sort_formats(formats) | ||||
|  | ||||
|             effTitle = showId + '-' + epTitle + ' part ' + compat_str(partNum+1) | ||||
|             results.append({ | ||||
|                 'id': shortMediaId, | ||||
|             virtual_id = show_name + ' ' + epTitle + ' part ' + compat_str(part_num + 1) | ||||
|             entries.append({ | ||||
|                 'id': guid, | ||||
|                 'title': virtual_id, | ||||
|                 'formats': formats, | ||||
|                 'uploader': showId, | ||||
|                 'upload_date': officialDate, | ||||
|                 'title': effTitle, | ||||
|                 'thumbnail': None, | ||||
|                 'description': compat_str(officialTitle), | ||||
|                 'uploader': show_name, | ||||
|                 'upload_date': upload_date, | ||||
|                 'duration': duration, | ||||
|                 'thumbnail': thumbnail, | ||||
|                 'description': description, | ||||
|             }) | ||||
|  | ||||
|         return results | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|             'entries': entries, | ||||
|             'title': show_name + ' ' + title, | ||||
|             'description': description, | ||||
|         } | ||||
|   | ||||
| @@ -74,7 +74,7 @@ class InfoExtractor(object): | ||||
|                                  "http", "https", "rtsp", "rtmp", "m3u8" or so. | ||||
|                     * preference Order number of this format. If this field is | ||||
|                                  present and not None, the formats get sorted | ||||
|                                  by this field. | ||||
|                                  by this field, regardless of all other values. | ||||
|                                  -1 for default (order by other properties), | ||||
|                                  -2 or smaller for less than default. | ||||
|                     * quality    Order number of the video quality of this | ||||
| @@ -92,8 +92,12 @@ class InfoExtractor(object): | ||||
|                     unique, but available before title. Typically, id is | ||||
|                     something like "4234987", title "Dancing naked mole rats", | ||||
|                     and display_id "dancing-naked-mole-rats" | ||||
|     thumbnails:     A list of dictionaries (with the entries "resolution" and | ||||
|                     "url") for the varying thumbnails | ||||
|     thumbnails:     A list of dictionaries, with the following entries: | ||||
|                         * "url" | ||||
|                         * "width" (optional, int) | ||||
|                         * "height" (optional, int) | ||||
|                         * "resolution" (optional, string "{width}x{height"}, | ||||
|                                         deprecated) | ||||
|     thumbnail:      Full URL to a video thumbnail image. | ||||
|     description:    One-line video description. | ||||
|     uploader:       Full name of the video uploader. | ||||
| @@ -113,6 +117,8 @@ class InfoExtractor(object): | ||||
|     webpage_url:    The url to the video webpage, if given to youtube-dl it | ||||
|                     should allow to get the same result again. (It will be set | ||||
|                     by YoutubeDL if it's missing) | ||||
|     categories:     A list of categories that the video falls in, for example | ||||
|                     ["Sports", "Berlin"] | ||||
|  | ||||
|     Unless mentioned otherwise, the fields should be Unicode strings. | ||||
|  | ||||
| @@ -242,16 +248,31 @@ class InfoExtractor(object): | ||||
|                 url = url_or_request.get_full_url() | ||||
|             except AttributeError: | ||||
|                 url = url_or_request | ||||
|             if len(url) > 200: | ||||
|                 h = u'___' + hashlib.md5(url.encode('utf-8')).hexdigest() | ||||
|                 url = url[:200 - len(h)] + h | ||||
|             raw_filename = ('%s_%s.dump' % (video_id, url)) | ||||
|             basen = '%s_%s' % (video_id, url) | ||||
|             if len(basen) > 240: | ||||
|                 h = u'___' + hashlib.md5(basen.encode('utf-8')).hexdigest() | ||||
|                 basen = basen[:240 - len(h)] + h | ||||
|             raw_filename = basen + '.dump' | ||||
|             filename = sanitize_filename(raw_filename, restricted=True) | ||||
|             self.to_screen(u'Saving request to ' + filename) | ||||
|             with open(filename, 'wb') as outf: | ||||
|                 outf.write(webpage_bytes) | ||||
|  | ||||
|         content = webpage_bytes.decode(encoding, 'replace') | ||||
|         try: | ||||
|             content = webpage_bytes.decode(encoding, 'replace') | ||||
|         except LookupError: | ||||
|             content = webpage_bytes.decode('utf-8', 'replace') | ||||
|  | ||||
|         if (u'<title>Access to this site is blocked</title>' in content and | ||||
|                 u'Websense' in content[:512]): | ||||
|             msg = u'Access to this webpage has been blocked by Websense filtering software in your network.' | ||||
|             blocked_iframe = self._html_search_regex( | ||||
|                 r'<iframe src="([^"]+)"', content, | ||||
|                 u'Websense information URL', default=None) | ||||
|             if blocked_iframe: | ||||
|                 msg += u' Visit %s for more details' % blocked_iframe | ||||
|             raise ExtractorError(msg, expected=True) | ||||
|  | ||||
|         return (content, urlh) | ||||
|  | ||||
|     def _download_webpage(self, url_or_request, video_id, note=None, errnote=None, fatal=True): | ||||
| @@ -265,9 +286,12 @@ class InfoExtractor(object): | ||||
|  | ||||
|     def _download_xml(self, url_or_request, video_id, | ||||
|                       note=u'Downloading XML', errnote=u'Unable to download XML', | ||||
|                       transform_source=None): | ||||
|                       transform_source=None, fatal=True): | ||||
|         """Return the xml as an xml.etree.ElementTree.Element""" | ||||
|         xml_string = self._download_webpage(url_or_request, video_id, note, errnote) | ||||
|         xml_string = self._download_webpage( | ||||
|             url_or_request, video_id, note, errnote, fatal=fatal) | ||||
|         if xml_string is False: | ||||
|             return xml_string | ||||
|         if transform_source: | ||||
|             xml_string = transform_source(xml_string) | ||||
|         return xml.etree.ElementTree.fromstring(xml_string.encode('utf-8')) | ||||
| @@ -435,6 +459,9 @@ class InfoExtractor(object): | ||||
|         if secure: regexes = self._og_regexes('video:secure_url') + regexes | ||||
|         return self._html_search_regex(regexes, html, name, **kargs) | ||||
|  | ||||
|     def _og_search_url(self, html, **kargs): | ||||
|         return self._og_search_property('url', html, **kargs) | ||||
|  | ||||
|     def _html_search_meta(self, name, html, display_name=None, fatal=False): | ||||
|         if display_name is None: | ||||
|             display_name = name | ||||
| @@ -531,6 +558,23 @@ class InfoExtractor(object): | ||||
|             ) | ||||
|         formats.sort(key=_formats_key) | ||||
|  | ||||
|     def http_scheme(self): | ||||
|         """ Either "https:" or "https:", depending on the user's preferences """ | ||||
|         return ( | ||||
|             'http:' | ||||
|             if self._downloader.params.get('prefer_insecure', False) | ||||
|             else 'https:') | ||||
|  | ||||
|     def _proto_relative_url(self, url, scheme=None): | ||||
|         if url is None: | ||||
|             return url | ||||
|         if url.startswith('//'): | ||||
|             if scheme is None: | ||||
|                 scheme = self.http_scheme() | ||||
|             return scheme + url | ||||
|         else: | ||||
|             return url | ||||
|  | ||||
|  | ||||
| class SearchInfoExtractor(InfoExtractor): | ||||
|     """ | ||||
| @@ -574,3 +618,4 @@ class SearchInfoExtractor(InfoExtractor): | ||||
|     @property | ||||
|     def SEARCH_KEY(self): | ||||
|         return self._SEARCH_KEY | ||||
|  | ||||
|   | ||||
| @@ -28,16 +28,18 @@ class CondeNastIE(InfoExtractor): | ||||
|         'glamour': 'Glamour', | ||||
|         'wmagazine': 'W Magazine', | ||||
|         'vanityfair': 'Vanity Fair', | ||||
|         'cnevids': 'Condé Nast', | ||||
|     } | ||||
|  | ||||
|     _VALID_URL = r'http://(video|www)\.(?P<site>%s)\.com/(?P<type>watch|series|video)/(?P<id>.+)' % '|'.join(_SITES.keys()) | ||||
|     _VALID_URL = r'http://(video|www|player)\.(?P<site>%s)\.com/(?P<type>watch|series|video|embed)/(?P<id>[^/?#]+)' % '|'.join(_SITES.keys()) | ||||
|     IE_DESC = 'Condé Nast media group: %s' % ', '.join(sorted(_SITES.values())) | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://video.wired.com/watch/3d-printed-speakers-lit-with-led', | ||||
|         'file': '5171b343c2b4c00dd0c1ccb3.mp4', | ||||
|         'md5': '1921f713ed48aabd715691f774c451f7', | ||||
|         'info_dict': { | ||||
|             'id': '5171b343c2b4c00dd0c1ccb3', | ||||
|             'ext': 'mp4', | ||||
|             'title': '3D Printed Speakers Lit With LED', | ||||
|             'description': 'Check out these beautiful 3D printed LED speakers.  You can\'t actually buy them, but LumiGeek is working on a board that will let you make you\'re own.', | ||||
|         } | ||||
| @@ -55,12 +57,16 @@ class CondeNastIE(InfoExtractor): | ||||
|         entries = [self.url_result(build_url(path), 'CondeNast') for path in paths] | ||||
|         return self.playlist_result(entries, playlist_title=title) | ||||
|  | ||||
|     def _extract_video(self, webpage): | ||||
|         description = self._html_search_regex([r'<div class="cne-video-description">(.+?)</div>', | ||||
|                                                r'<div class="video-post-content">(.+?)</div>', | ||||
|                                                ], | ||||
|                                               webpage, 'description', | ||||
|                                               fatal=False, flags=re.DOTALL) | ||||
|     def _extract_video(self, webpage, url_type): | ||||
|         if url_type != 'embed': | ||||
|             description = self._html_search_regex( | ||||
|                 [ | ||||
|                     r'<div class="cne-video-description">(.+?)</div>', | ||||
|                     r'<div class="video-post-content">(.+?)</div>', | ||||
|                 ], | ||||
|                 webpage, 'description', fatal=False, flags=re.DOTALL) | ||||
|         else: | ||||
|             description = None | ||||
|         params = self._search_regex(r'var params = {(.+?)}[;,]', webpage, | ||||
|                                     'player params', flags=re.DOTALL) | ||||
|         video_id = self._search_regex(r'videoId: [\'"](.+?)[\'"]', params, 'video id') | ||||
| @@ -99,12 +105,12 @@ class CondeNastIE(InfoExtractor): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         site = mobj.group('site') | ||||
|         url_type = mobj.group('type') | ||||
|         id = mobj.group('id') | ||||
|         item_id = mobj.group('id') | ||||
|  | ||||
|         self.to_screen(u'Extracting from %s with the Condé Nast extractor' % self._SITES[site]) | ||||
|         webpage = self._download_webpage(url, id) | ||||
|         self.to_screen('Extracting from %s with the Condé Nast extractor' % self._SITES[site]) | ||||
|         webpage = self._download_webpage(url, item_id) | ||||
|  | ||||
|         if url_type == 'series': | ||||
|             return self._extract_series(url, webpage) | ||||
|         else: | ||||
|             return self._extract_video(webpage) | ||||
|             return self._extract_video(webpage, url_type) | ||||
|   | ||||
| @@ -4,6 +4,7 @@ import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     int_or_none, | ||||
|     unescapeHTML, | ||||
|     find_xpath_attr, | ||||
| ) | ||||
| @@ -54,18 +55,29 @@ class CSpanIE(InfoExtractor): | ||||
|         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) | ||||
|  | ||||
|         url = unescapeHTML(data['video']['files'][0]['path']['#text']) | ||||
|  | ||||
|         doc = self._download_xml('http://www.c-span.org/common/services/flashXml.php?programid=' + video_id, | ||||
|         doc = self._download_xml( | ||||
|             'http://www.c-span.org/common/services/flashXml.php?programid=' + video_id, | ||||
|             video_id) | ||||
|  | ||||
|         def find_string(s): | ||||
|             return find_xpath_attr(doc, './/string', 'name', s).text | ||||
|         title = find_xpath_attr(doc, './/string', 'name', 'title').text | ||||
|         thumbnail = find_xpath_attr(doc, './/string', 'name', 'poster').text | ||||
|  | ||||
|         files = data['video']['files'] | ||||
|  | ||||
|         entries = [{ | ||||
|             'id': '%s_%d' % (video_id, partnum + 1), | ||||
|             'title': ( | ||||
|                 title if len(files) == 1 else | ||||
|                 '%s part %d' % (title, partnum + 1)), | ||||
|             'url': unescapeHTML(f['path']['#text']), | ||||
|             'description': description, | ||||
|             'thumbnail': thumbnail, | ||||
|             'duration': int_or_none(f.get('length', {}).get('#text')), | ||||
|         } for partnum, f in enumerate(files)] | ||||
|  | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|             'entries': entries, | ||||
|             'title': title, | ||||
|             'id': video_id, | ||||
|             'title': find_string('title'), | ||||
|             'url': url, | ||||
|             'description': description, | ||||
|             'thumbnail': find_string('poster'), | ||||
|         } | ||||
|   | ||||
| @@ -8,13 +8,11 @@ from .subtitles import SubtitlesInfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_request, | ||||
|     compat_str, | ||||
|     get_element_by_attribute, | ||||
|     get_element_by_id, | ||||
|     orderedSet, | ||||
|     str_to_int, | ||||
|     int_or_none, | ||||
|  | ||||
|     ExtractorError, | ||||
|     unescapeHTML, | ||||
| ) | ||||
|  | ||||
| class DailymotionBaseInfoExtractor(InfoExtractor): | ||||
| @@ -152,7 +150,7 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|         return { | ||||
|             'id':       video_id, | ||||
|             'formats': formats, | ||||
|             'uploader': info['owner_screenname'], | ||||
|             'uploader': info['owner.screenname'], | ||||
|             'upload_date':  video_upload_date, | ||||
|             'title':    self._og_search_title(webpage), | ||||
|             'subtitles':    video_subtitles, | ||||
| @@ -180,7 +178,7 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
| class DailymotionPlaylistIE(DailymotionBaseInfoExtractor): | ||||
|     IE_NAME = u'dailymotion:playlist' | ||||
|     _VALID_URL = r'(?:https?://)?(?:www\.)?dailymotion\.[a-z]{2,3}/playlist/(?P<id>.+?)/' | ||||
|     _MORE_PAGES_INDICATOR = r'<div class="next">.*?<a.*?href="/playlist/.+?".*?>.*?</a>.*?</div>' | ||||
|     _MORE_PAGES_INDICATOR = r'(?s)<div class="pages[^"]*">.*?<a\s+class="[^"]*?icon-arrow_right[^"]*?"' | ||||
|     _PAGE_TEMPLATE = 'https://www.dailymotion.com/playlist/%s/%s' | ||||
|  | ||||
|     def _extract_entries(self, id): | ||||
| @@ -190,10 +188,9 @@ class DailymotionPlaylistIE(DailymotionBaseInfoExtractor): | ||||
|             webpage = self._download_webpage(request, | ||||
|                                              id, u'Downloading page %s' % pagenum) | ||||
|  | ||||
|             playlist_el = get_element_by_attribute(u'class', u'row video_list', webpage) | ||||
|             video_ids.extend(re.findall(r'data-id="(.+?)"', playlist_el)) | ||||
|             video_ids.extend(re.findall(r'data-xid="(.+?)"', webpage)) | ||||
|  | ||||
|             if re.search(self._MORE_PAGES_INDICATOR, webpage, re.DOTALL) is None: | ||||
|             if re.search(self._MORE_PAGES_INDICATOR, webpage) is None: | ||||
|                 break | ||||
|         return [self.url_result('http://www.dailymotion.com/video/%s' % video_id, 'Dailymotion') | ||||
|                    for video_id in orderedSet(video_ids)] | ||||
| @@ -203,26 +200,26 @@ class DailymotionPlaylistIE(DailymotionBaseInfoExtractor): | ||||
|         playlist_id = mobj.group('id') | ||||
|         webpage = self._download_webpage(url, playlist_id) | ||||
|  | ||||
|         return {'_type': 'playlist', | ||||
|                 'id': playlist_id, | ||||
|                 'title': get_element_by_id(u'playlist_name', webpage), | ||||
|                 'entries': self._extract_entries(playlist_id), | ||||
|                 } | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|             'id': playlist_id, | ||||
|             'title': self._og_search_title(webpage), | ||||
|             'entries': self._extract_entries(playlist_id), | ||||
|         } | ||||
|  | ||||
|  | ||||
| class DailymotionUserIE(DailymotionPlaylistIE): | ||||
|     IE_NAME = u'dailymotion:user' | ||||
|     _VALID_URL = r'(?:https?://)?(?:www\.)?dailymotion\.[a-z]{2,3}/user/(?P<user>[^/]+)' | ||||
|     _MORE_PAGES_INDICATOR = r'<div class="next">.*?<a.*?href="/user/.+?".*?>.*?</a>.*?</div>' | ||||
|     _VALID_URL = r'https?://(?:www\.)?dailymotion\.[a-z]{2,3}/user/(?P<user>[^/]+)' | ||||
|     _PAGE_TEMPLATE = 'http://www.dailymotion.com/user/%s/%s' | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         user = mobj.group('user') | ||||
|         webpage = self._download_webpage(url, user) | ||||
|         full_user = self._html_search_regex( | ||||
|             r'<a class="label" href="/%s".*?>(.*?)</' % re.escape(user), | ||||
|             webpage, u'user', flags=re.DOTALL) | ||||
|         full_user = unescapeHTML(self._html_search_regex( | ||||
|             r'<a class="nav-image" title="([^"]+)" href="/%s">' % re.escape(user), | ||||
|             webpage, u'user', flags=re.DOTALL)) | ||||
|  | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|   | ||||
| @@ -7,12 +7,13 @@ from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class DiscoveryIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://dsc\.discovery\.com\/[a-zA-Z0-9\-]*/[a-zA-Z0-9\-]*/videos/(?P<id>[a-zA-Z0-9\-]*)(.htm)?' | ||||
|     _VALID_URL = r'http://www\.discovery\.com\/[a-zA-Z0-9\-]*/[a-zA-Z0-9\-]*/videos/(?P<id>[a-zA-Z0-9\-]*)(.htm)?' | ||||
|     _TEST = { | ||||
|         'url': 'http://dsc.discovery.com/tv-shows/mythbusters/videos/mission-impossible-outtakes.htm', | ||||
|         'file': '614784.mp4', | ||||
|         'url': 'http://www.discovery.com/tv-shows/mythbusters/videos/mission-impossible-outtakes.htm', | ||||
|         'md5': 'e12614f9ee303a6ccef415cb0793eba2', | ||||
|         'info_dict': { | ||||
|             'id': '614784', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'MythBusters: Mission Impossible Outtakes', | ||||
|             'description': ('Watch Jamie Hyneman and Adam Savage practice being' | ||||
|                 ' each other -- to the point of confusing Jamie\'s dog -- and ' | ||||
| @@ -34,7 +35,7 @@ class DiscoveryIE(InfoExtractor): | ||||
|         formats = [] | ||||
|         for f in info['mp4']: | ||||
|             formats.append( | ||||
|                 {'url': f['src'], r'ext': r'mp4', 'tbr': int(f['bitrate'][:-1])}) | ||||
|                 {'url': f['src'], 'ext': 'mp4', 'tbr': int(f['bitrate'][:-1])}) | ||||
|  | ||||
|         return { | ||||
|             'id': info['contentId'], | ||||
|   | ||||
							
								
								
									
										27
									
								
								youtube_dl/extractor/divxstage.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								youtube_dl/extractor/divxstage.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from .novamov import NovaMovIE | ||||
|  | ||||
|  | ||||
| class DivxStageIE(NovaMovIE): | ||||
|     IE_NAME = 'divxstage' | ||||
|     IE_DESC = 'DivxStage' | ||||
|  | ||||
|     _VALID_URL = NovaMovIE._VALID_URL_TEMPLATE % {'host': 'divxstage\.(?:eu|net|ch|co|at|ag)'} | ||||
|  | ||||
|     _HOST = 'www.divxstage.eu' | ||||
|  | ||||
|     _FILE_DELETED_REGEX = r'>This file no longer exists on our servers.<' | ||||
|     _TITLE_REGEX = r'<div class="video_det">\s*<strong>([^<]+)</strong>' | ||||
|     _DESCRIPTION_REGEX = r'<div class="video_det">\s*<strong>[^<]+</strong>\s*<p>([^<]+)</p>' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.divxstage.eu/video/57f238e2e5e01', | ||||
|         'md5': '63969f6eb26533a1968c4d325be63e72', | ||||
|         'info_dict': { | ||||
|             'id': '57f238e2e5e01', | ||||
|             'ext': 'flv', | ||||
|             'title': 'youtubedl test video', | ||||
|             'description': 'This is a test video for youtubedl.', | ||||
|         } | ||||
|     } | ||||
| @@ -1,39 +1,37 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     unified_strdate, | ||||
| ) | ||||
| from ..utils import 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 = { | ||||
|         u"url": u"http://www.3sat.de/mediathek/index.php?obj=36983", | ||||
|         u'file': u'36983.mp4', | ||||
|         u'md5': u'9dcfe344732808dbfcc901537973c922', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Kaffeeland Schweiz", | ||||
|             u"description": u"Über 80 Kaffeeröstereien liefern in der Schweiz das Getränk, in das das Land so vernarrt ist: Mehr als 1000 Tassen trinkt ein Schweizer pro Jahr. SCHWEIZWEIT nimmt die Kaffeekultur unter die...",  | ||||
|             u"uploader": u"3sat", | ||||
|             u"upload_date": u"20130622" | ||||
|         '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' | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         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, note=u'Downloading video details') | ||||
|         details_doc = self._download_xml(details_url, video_id, 'Downloading video details') | ||||
|  | ||||
|         thumbnail_els = details_doc.findall('.//teaserimage') | ||||
|         thumbnails = [{ | ||||
|             'width': te.attrib['key'].partition('x')[0], | ||||
|             'height': te.attrib['key'].partition('x')[2], | ||||
|             'width': int(te.attrib['key'].partition('x')[0]), | ||||
|             'height': int(te.attrib['key'].partition('x')[2]), | ||||
|             'url': te.text, | ||||
|         } for te in thumbnail_els] | ||||
|  | ||||
|   | ||||
| @@ -1,23 +1,25 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse, | ||||
|     determine_ext | ||||
| ) | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class EHowIE(InfoExtractor): | ||||
|     IE_NAME = u'eHow' | ||||
|     _VALID_URL = r'(?:https?://)?(?:www\.)?ehow\.com/[^/_?]*_(?P<id>[0-9]+)' | ||||
|     IE_NAME = 'eHow' | ||||
|     _VALID_URL = r'https?://(?:www\.)?ehow\.com/[^/_?]*_(?P<id>[0-9]+)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.ehow.com/video_12245069_hardwood-flooring-basics.html', | ||||
|         u'file': u'12245069.flv', | ||||
|         u'md5': u'9809b4e3f115ae2088440bcb4efbf371', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Hardwood Flooring Basics", | ||||
|             u"description": u"Hardwood flooring may be time consuming, but its ultimately a pretty straightforward concept. Learn about hardwood flooring basics with help from a hardware flooring business owner in this free video...", | ||||
|    			u"uploader": u"Erick Nathan" | ||||
|         'url': 'http://www.ehow.com/video_12245069_hardwood-flooring-basics.html', | ||||
|         'md5': '9809b4e3f115ae2088440bcb4efbf371', | ||||
|         'info_dict': { | ||||
|             'id': '12245069', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Hardwood Flooring Basics', | ||||
|             'description': 'Hardwood flooring may be time consuming, but its ultimately a pretty straightforward concept. Learn about hardwood flooring basics with help from a hardware flooring business owner in this free video...', | ||||
|             'uploader': 'Erick Nathan', | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -26,21 +28,16 @@ class EHowIE(InfoExtractor): | ||||
|         video_id = mobj.group('id') | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         video_url = self._search_regex(r'(?:file|source)=(http[^\'"&]*)', | ||||
|             webpage, u'video URL') | ||||
|         final_url = compat_urllib_parse.unquote(video_url)         | ||||
|         uploader = self._search_regex(r'<meta name="uploader" content="(.+?)" />', | ||||
|             webpage, u'uploader') | ||||
|             webpage, 'video URL') | ||||
|         final_url = compat_urllib_parse.unquote(video_url) | ||||
|         uploader = self._html_search_meta('uploader', webpage) | ||||
|         title = self._og_search_title(webpage).replace(' | eHow', '') | ||||
|         ext = determine_ext(final_url) | ||||
|  | ||||
|         return { | ||||
|             '_type':       'video', | ||||
|             'id':          video_id, | ||||
|             'url':         final_url, | ||||
|             'ext':         ext, | ||||
|             'title':       title, | ||||
|             'thumbnail':   self._og_search_thumbnail(webpage), | ||||
|             'id': video_id, | ||||
|             'url': final_url, | ||||
|             'title': title, | ||||
|             'thumbnail': self._og_search_thumbnail(webpage), | ||||
|             'description': self._og_search_description(webpage), | ||||
|             'uploader':    uploader, | ||||
|             'uploader': uploader, | ||||
|         } | ||||
|  | ||||
|   | ||||
							
								
								
									
										54
									
								
								youtube_dl/extractor/empflix.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								youtube_dl/extractor/empflix.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class EmpflixIE(InfoExtractor): | ||||
|     _VALID_URL = r'^https?://www\.empflix\.com/videos/.*?-(?P<id>[0-9]+)\.html' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.empflix.com/videos/Amateur-Finger-Fuck-33051.html', | ||||
|         'md5': 'b1bc15b6412d33902d6e5952035fcabc', | ||||
|         'info_dict': { | ||||
|             'id': '33051', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Amateur Finger Fuck', | ||||
|             'description': 'Amateur solo finger fucking.', | ||||
|             'age_limit': 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         age_limit = self._rta_search(webpage) | ||||
|  | ||||
|         video_title = self._html_search_regex( | ||||
|             r'name="title" value="(?P<title>[^"]*)"', webpage, 'title') | ||||
|         video_description = self._html_search_regex( | ||||
|             r'name="description" value="([^"]*)"', webpage, 'description', fatal=False) | ||||
|  | ||||
|         cfg_url = self._html_search_regex( | ||||
|             r'flashvars\.config = escape\("([^"]+)"', | ||||
|             webpage, 'flashvars.config') | ||||
|  | ||||
|         cfg_xml = self._download_xml( | ||||
|             cfg_url, video_id, note='Downloading metadata') | ||||
|  | ||||
|         formats = [ | ||||
|             { | ||||
|                 'url': item.find('videoLink').text, | ||||
|                 'format_id': item.find('res').text, | ||||
|             } for item in cfg_xml.findall('./quality/item') | ||||
|         ] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': video_title, | ||||
|             'description': video_description, | ||||
|             'formats': formats, | ||||
|             'age_limit': age_limit, | ||||
|         } | ||||
| @@ -1,4 +1,5 @@ | ||||
| import os | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| @@ -8,18 +9,23 @@ from ..utils import ( | ||||
|     compat_urllib_parse, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class ExtremeTubeIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>extremetube\.com/video/.+?(?P<videoid>[0-9]+))(?:[/?&]|$)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.extremetube.com/video/music-video-14-british-euro-brit-european-cumshots-swallow-652431', | ||||
|         u'file': u'652431.mp4', | ||||
|         u'md5': u'1fb9228f5e3332ec8c057d6ac36f33e0', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Music Video 14 british euro brit european cumshots swallow", | ||||
|             u"uploader": u"unknown", | ||||
|             u"age_limit": 18, | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>extremetube\.com/.*?video/.+?(?P<videoid>[0-9]+))(?:[/?&]|$)' | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.extremetube.com/video/music-video-14-british-euro-brit-european-cumshots-swallow-652431', | ||||
|         'md5': '1fb9228f5e3332ec8c057d6ac36f33e0', | ||||
|         'info_dict': { | ||||
|             'id': '652431', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Music Video 14 british euro brit european cumshots swallow', | ||||
|             'uploader': 'unknown', | ||||
|             'age_limit': 18, | ||||
|         } | ||||
|     } | ||||
|     }, { | ||||
|         'url': 'http://www.extremetube.com/gay/video/abcde-1234', | ||||
|         'only_matching': True, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
| @@ -30,11 +36,14 @@ class ExtremeTubeIE(InfoExtractor): | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<h1 [^>]*?title="([^"]+)"[^>]*>\1<', webpage, u'title') | ||||
|         uploader = self._html_search_regex(r'>Posted by:(?=<)(?:\s|<[^>]*>)*(.+?)\|', webpage, u'uploader', fatal=False) | ||||
|         video_url = compat_urllib_parse.unquote(self._html_search_regex(r'video_url=(.+?)&', webpage, u'video_url')) | ||||
|         video_title = self._html_search_regex( | ||||
|             r'<h1 [^>]*?title="([^"]+)"[^>]*>', webpage, 'title') | ||||
|         uploader = self._html_search_regex( | ||||
|             r'>Posted by:(?=<)(?:\s|<[^>]*>)*(.+?)\|', webpage, 'uploader', | ||||
|             fatal=False) | ||||
|         video_url = compat_urllib_parse.unquote(self._html_search_regex( | ||||
|             r'video_url=(.+?)&', webpage, 'video_url')) | ||||
|         path = compat_urllib_parse_urlparse(video_url).path | ||||
|         extension = os.path.splitext(path)[1][1:] | ||||
|         format = path.split('/')[5].split('_')[:2] | ||||
|         format = "-".join(format) | ||||
|  | ||||
| @@ -43,7 +52,6 @@ class ExtremeTubeIE(InfoExtractor): | ||||
|             'title': video_title, | ||||
|             'uploader': uploader, | ||||
|             'url': video_url, | ||||
|             'ext': extension, | ||||
|             'format': format, | ||||
|             'format_id': format, | ||||
|             'age_limit': 18, | ||||
|   | ||||
| @@ -76,9 +76,8 @@ class FacebookIE(InfoExtractor): | ||||
|  | ||||
|             check_form = { | ||||
|                 'fb_dtsg': self._search_regex(r'name="fb_dtsg" value="(.+?)"', login_results, 'fb_dtsg'), | ||||
|                 'nh': self._search_regex(r'name="nh" value="(\w*?)"', login_results, 'nh'), | ||||
|                 'h': self._search_regex(r'name="h" value="(\w*?)"', login_results, 'h'), | ||||
|                 'name_action_selected': 'dont_save', | ||||
|                 'submit[Continue]': self._search_regex(r'<button[^>]+value="(.*?)"[^>]+name="submit\[Continue\]"', login_results, 'continue'), | ||||
|             } | ||||
|             check_req = compat_urllib_request.Request(self._CHECKPOINT_URL, urlencode_postdata(check_form)) | ||||
|             check_req.add_header('Content-Type', 'application/x-www-form-urlencoded') | ||||
|   | ||||
							
								
								
									
										63
									
								
								youtube_dl/extractor/fc2.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								youtube_dl/extractor/fc2.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| #! -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import hashlib | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     compat_urllib_request, | ||||
|     compat_urlparse, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class FC2IE(InfoExtractor): | ||||
|     _VALID_URL = r'^http://video\.fc2\.com/((?P<lang>[^/]+)/)?content/(?P<id>[^/]+)' | ||||
|     IE_NAME = 'fc2' | ||||
|     _TEST = { | ||||
|         'url': 'http://video.fc2.com/en/content/20121103kUan1KHs', | ||||
|         'md5': 'a6ebe8ebe0396518689d963774a54eb7', | ||||
|         'info_dict': { | ||||
|             'id': '20121103kUan1KHs', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Boxing again with Puff', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         self._downloader.cookiejar.clear_session_cookies()  # must clear | ||||
|  | ||||
|         title = self._og_search_title(webpage) | ||||
|         thumbnail = self._og_search_thumbnail(webpage) | ||||
|         refer = url.replace('/content/', '/a/content/') | ||||
|  | ||||
|         mimi = hashlib.md5((video_id + '_gGddgPfeaf_gzyr').encode('utf-8')).hexdigest() | ||||
|  | ||||
|         info_url = ( | ||||
|             "http://video.fc2.com/ginfo.php?mimi={1:s}&href={2:s}&v={0:s}&fversion=WIN%2011%2C6%2C602%2C180&from=2&otag=0&upid={0:s}&tk=null&". | ||||
|             format(video_id, mimi, compat_urllib_request.quote(refer, safe='').replace('.','%2E'))) | ||||
|  | ||||
|         info_webpage = self._download_webpage( | ||||
|             info_url, video_id, note='Downloading info page') | ||||
|         info = compat_urlparse.parse_qs(info_webpage) | ||||
|  | ||||
|         if 'err_code' in info: | ||||
|             raise ExtractorError('Error code: %s' % info['err_code'][0]) | ||||
|  | ||||
|         video_url = info['filepath'][0] + '?mid=' + info['mid'][0] | ||||
|         title_info = info.get('title') | ||||
|         if title_info: | ||||
|             title = title_info[0] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'url': video_url, | ||||
|             'ext': 'flv', | ||||
|             'thumbnail': thumbnail, | ||||
|         } | ||||
| @@ -6,7 +6,6 @@ from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class FirstpostIE(InfoExtractor): | ||||
|     IE_NAME = 'Firstpost.com' | ||||
|     _VALID_URL = r'http://(?:www\.)?firstpost\.com/[^/]+/.*-(?P<id>[0-9]+)\.html' | ||||
|  | ||||
|     _TEST = { | ||||
| @@ -16,7 +15,7 @@ class FirstpostIE(InfoExtractor): | ||||
|             'id': '1025403', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'India to launch indigenous aircraft carrier INS Vikrant today', | ||||
|             'description': 'Its flight deck is over twice the size of a football field, its power unit can light up the entire Kochi city and the cabling is enough to cover the distance between here to Delhi.', | ||||
|             'description': 'md5:feef3041cb09724e0bdc02843348f5f4', | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -24,15 +23,30 @@ class FirstpostIE(InfoExtractor): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         video_url = self._html_search_regex( | ||||
|             r'<div.*?name="div_video".*?flashvars="([^"]+)">', | ||||
|             webpage, 'video URL') | ||||
|         page = self._download_webpage(url, video_id) | ||||
|         title = self._html_search_meta('twitter:title', page, 'title') | ||||
|         description = self._html_search_meta('twitter:description', page, 'title') | ||||
|  | ||||
|         data = self._download_xml( | ||||
|             'http://www.firstpost.com/getvideoxml-%s.xml' % video_id, video_id, | ||||
|             'Downloading video XML') | ||||
|  | ||||
|         item = data.find('./playlist/item') | ||||
|         thumbnail = item.find('./image').text | ||||
|  | ||||
|         formats = [ | ||||
|             { | ||||
|                 'url': details.find('./file').text, | ||||
|                 'format_id': details.find('./label').text.strip(), | ||||
|                 'width': int(details.find('./width').text.strip()), | ||||
|                 'height': int(details.find('./height').text.strip()), | ||||
|             } for details in item.findall('./source/file_details') if details.find('./file').text | ||||
|         ] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'title': self._og_search_title(webpage), | ||||
|             'description': self._og_search_description(webpage), | ||||
|             'thumbnail': self._og_search_thumbnail(webpage), | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'thumbnail': thumbnail, | ||||
|             'formats': formats, | ||||
|         } | ||||
|   | ||||
| @@ -5,6 +5,8 @@ import re | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_str, | ||||
|     compat_urllib_parse, | ||||
|     ExtractorError, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -16,16 +18,28 @@ class FiveMinIE(InfoExtractor): | ||||
|         (?P<id>\d+) | ||||
|         ''' | ||||
|  | ||||
|     _TEST = { | ||||
|         # From http://www.engadget.com/2013/11/15/ipad-mini-retina-display-review/ | ||||
|         'url': 'http://pshared.5min.com/Scripts/PlayerSeed.js?sid=281&width=560&height=345&playList=518013791', | ||||
|         'md5': '4f7b0b79bf1a470e5004f7112385941d', | ||||
|         'info_dict': { | ||||
|             'id': '518013791', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'iPad Mini with Retina Display Review', | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             # From http://www.engadget.com/2013/11/15/ipad-mini-retina-display-review/ | ||||
|             'url': 'http://pshared.5min.com/Scripts/PlayerSeed.js?sid=281&width=560&height=345&playList=518013791', | ||||
|             'md5': '4f7b0b79bf1a470e5004f7112385941d', | ||||
|             'info_dict': { | ||||
|                 'id': '518013791', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'iPad Mini with Retina Display Review', | ||||
|             }, | ||||
|         }, | ||||
|     } | ||||
|         { | ||||
|             # From http://on.aol.com/video/how-to-make-a-next-level-fruit-salad-518086247 | ||||
|             'url': '5min:518086247', | ||||
|             'md5': 'e539a9dd682c288ef5a498898009f69e', | ||||
|             'info_dict': { | ||||
|                 'id': '518086247', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'How to Make a Next-Level Fruit Salad', | ||||
|             }, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     @classmethod | ||||
|     def _build_result(cls, video_id): | ||||
| @@ -34,10 +48,28 @@ class FiveMinIE(InfoExtractor): | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         info = self._download_json( | ||||
|             'https://syn.5min.com/handlers/SenseHandler.ashx?func=GetResults&' | ||||
|             'playlist=%s&url=https' % video_id, | ||||
|             video_id)['binding'][0] | ||||
|         embed_url = 'https://embed.5min.com/playerseed/?playList=%s' % video_id | ||||
|         embed_page = self._download_webpage(embed_url, video_id, | ||||
|             'Downloading embed page') | ||||
|         sid = self._search_regex(r'sid=(\d+)', embed_page, 'sid') | ||||
|         query = compat_urllib_parse.urlencode({ | ||||
|             'func': 'GetResults', | ||||
|             'playlist': video_id, | ||||
|             'sid': sid, | ||||
|             'isPlayerSeed': 'true', | ||||
|             'url': embed_url, | ||||
|         }) | ||||
|         response = self._download_json( | ||||
|             'https://syn.5min.com/handlers/SenseHandler.ashx?' + query, | ||||
|             video_id) | ||||
|         if not response['success']: | ||||
|             err_msg = response['errorMessage'] | ||||
|             if err_msg == 'ErrorVideoUserNotGeo': | ||||
|                 msg = 'Video not available from your location' | ||||
|             else: | ||||
|                 msg = 'Aol said: %s' % err_msg | ||||
|             raise ExtractorError(msg, expected=True, video_id=video_id) | ||||
|         info = response['binding'][0] | ||||
|  | ||||
|         second_id = compat_str(int(video_id[:-2]) + 1) | ||||
|         formats = [] | ||||
|   | ||||
							
								
								
									
										77
									
								
								youtube_dl/extractor/franceculture.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								youtube_dl/extractor/franceculture.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import json | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_parse_qs, | ||||
|     compat_urlparse, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class FranceCultureIE(InfoExtractor): | ||||
|     _VALID_URL = r'(?P<baseurl>http://(?:www\.)?franceculture\.fr/)player/reecouter\?play=(?P<id>[0-9]+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.franceculture.fr/player/reecouter?play=4795174', | ||||
|         'info_dict': { | ||||
|             'id': '4795174', | ||||
|             'ext': 'mp3', | ||||
|             'title': 'Rendez-vous au pays des geeks', | ||||
|             'vcodec': 'none', | ||||
|             'uploader': 'Colette Fellous', | ||||
|             'upload_date': '20140301', | ||||
|             'duration': 3601, | ||||
|             'thumbnail': r're:^http://www\.franceculture\.fr/.*/images/player/Carnet-nomade\.jpg$', | ||||
|             'description': 'Avec :Jean-Baptiste Péretié pour son documentaire sur Arte "La revanche des « geeks », une enquête menée aux Etats-Unis dans la S ...', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         baseurl = mobj.group('baseurl') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         params_code = self._search_regex( | ||||
|             r"<param name='movie' value='/sites/all/modules/rf/rf_player/swf/loader.swf\?([^']+)' />", | ||||
|             webpage, 'parameter code') | ||||
|         params = compat_parse_qs(params_code) | ||||
|         video_url = compat_urlparse.urljoin(baseurl, params['urlAOD'][0]) | ||||
|  | ||||
|         title = self._html_search_regex( | ||||
|             r'<h1 class="title[^"]+">(.+?)</h1>', webpage, 'title') | ||||
|         uploader = self._html_search_regex( | ||||
|             r'(?s)<div id="emission".*?<span class="author">(.*?)</span>', | ||||
|             webpage, 'uploader', fatal=False) | ||||
|         thumbnail_part = self._html_search_regex( | ||||
|             r'(?s)<div id="emission".*?<img src="([^"]+)"', webpage, | ||||
|             'thumbnail', fatal=False) | ||||
|         if thumbnail_part is None: | ||||
|             thumbnail = None | ||||
|         else: | ||||
|             thumbnail = compat_urlparse.urljoin(baseurl, thumbnail_part) | ||||
|         description = self._html_search_regex( | ||||
|             r'(?s)<p class="desc">(.*?)</p>', webpage, 'description') | ||||
|  | ||||
|         info = json.loads(params['infoData'][0])[0] | ||||
|         duration = info.get('media_length') | ||||
|         upload_date_candidate = info.get('media_section5') | ||||
|         upload_date = ( | ||||
|             upload_date_candidate | ||||
|             if (upload_date_candidate is not None and | ||||
|                 re.match(r'[0-9]{8}$', upload_date_candidate)) | ||||
|             else None) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'vcodec': 'none' if video_url.lower().endswith('.mp3') else None, | ||||
|             'duration': duration, | ||||
|             'uploader': uploader, | ||||
|             'upload_date': upload_date, | ||||
|             'title': title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'description': description, | ||||
|         } | ||||
| @@ -48,24 +48,36 @@ class PluzzIE(FranceTVBaseInfoExtractor): | ||||
|  | ||||
| class FranceTvInfoIE(FranceTVBaseInfoExtractor): | ||||
|     IE_NAME = 'francetvinfo.fr' | ||||
|     _VALID_URL = r'https?://www\.francetvinfo\.fr/replay.*/(?P<title>.+)\.html' | ||||
|     _VALID_URL = r'https?://www\.francetvinfo\.fr/.*/(?P<title>.+)\.html' | ||||
|  | ||||
|     _TEST = { | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.francetvinfo.fr/replay-jt/france-3/soir-3/jt-grand-soir-3-lundi-26-aout-2013_393427.html', | ||||
|         'file': '84981923.mp4', | ||||
|         'info_dict': { | ||||
|             'id': '84981923', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Soir 3', | ||||
|         }, | ||||
|         'params': { | ||||
|             'skip_download': True, | ||||
|         }, | ||||
|     } | ||||
|     }, { | ||||
|         'url': 'http://www.francetvinfo.fr/elections/europeennes/direct-europeennes-regardez-le-debat-entre-les-candidats-a-la-presidence-de-la-commission_600639.html', | ||||
|         'info_dict': { | ||||
|             'id': 'EV_20019', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Débat des candidats à la Commission européenne', | ||||
|             'description': 'Débat des candidats à la Commission européenne', | ||||
|         }, | ||||
|         'params': { | ||||
|             'skip_download': 'HLS (reqires ffmpeg)' | ||||
|         } | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         page_title = mobj.group('title') | ||||
|         webpage = self._download_webpage(url, page_title) | ||||
|         video_id = self._search_regex(r'id-video=(\d+?)[@"]', webpage, 'video id') | ||||
|         video_id = self._search_regex(r'id-video=((?:[^0-9]*?_)?[0-9]+)[@"]', webpage, 'video id') | ||||
|         return self._extract_video(video_id) | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -4,22 +4,32 @@ import json | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ExtractorError | ||||
|  | ||||
|  | ||||
| class FunnyOrDieIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?funnyordie\.com/(?P<type>embed|videos)/(?P<id>[0-9a-f]+)(?:$|[?#/])' | ||||
|     _TEST = { | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.funnyordie.com/videos/0732f586d7/heart-shaped-box-literal-video-version', | ||||
|         'file': '0732f586d7.mp4', | ||||
|         'md5': 'f647e9e90064b53b6e046e75d0241fbd', | ||||
|         'md5': 'bcd81e0c4f26189ee09be362ad6e6ba9', | ||||
|         'info_dict': { | ||||
|             'description': ('Lyrics changed to match the video. Spoken cameo ' | ||||
|                 'by Obscurus Lupa (from ThatGuyWithTheGlasses.com). Based on a ' | ||||
|                 'concept by Dustin McLean (DustFilms.com). Performed, edited, ' | ||||
|                 'and written by David A. Scott.'), | ||||
|             'id': '0732f586d7', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Heart-Shaped Box: Literal Video Version', | ||||
|             'description': 'md5:ea09a01bc9a1c46d9ab696c01747c338', | ||||
|             'thumbnail': 're:^http:.*\.jpg$', | ||||
|         }, | ||||
|     } | ||||
|     }, { | ||||
|         'url': 'http://www.funnyordie.com/embed/e402820827', | ||||
|         'md5': 'ff4d83318f89776ed0250634cfaa8d36', | ||||
|         'info_dict': { | ||||
|             'id': 'e402820827', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Please Use This Song (Jon Lajoie)', | ||||
|             'description': 'md5:2ed27d364f5a805a6dba199faaf6681d', | ||||
|             'thumbnail': 're:^http:.*\.jpg$', | ||||
|         }, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
| @@ -27,27 +37,34 @@ class FunnyOrDieIE(InfoExtractor): | ||||
|         video_id = mobj.group('id') | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         video_url = self._search_regex( | ||||
|             [r'type="video/mp4" src="(.*?)"', r'src="([^>]*?)" type=\'video/mp4\''], | ||||
|             webpage, 'video URL', flags=re.DOTALL) | ||||
|         links = re.findall(r'<source src="([^"]+/v)\d+\.([^"]+)" type=\'video', webpage) | ||||
|         if not links: | ||||
|             raise ExtractorError('No media links available for %s' % video_id) | ||||
|  | ||||
|         if mobj.group('type') == 'embed': | ||||
|             post_json = self._search_regex( | ||||
|                 r'fb_post\s*=\s*(\{.*?\});', webpage, 'post details') | ||||
|             post = json.loads(post_json) | ||||
|             title = post['name'] | ||||
|             description = post.get('description') | ||||
|             thumbnail = post.get('picture') | ||||
|         else: | ||||
|             title = self._og_search_title(webpage) | ||||
|             description = self._og_search_description(webpage) | ||||
|             thumbnail = None | ||||
|         links.sort(key=lambda link: 1 if link[1] == 'mp4' else 0) | ||||
|  | ||||
|         bitrates = self._html_search_regex(r'<source src="[^"]+/v,((?:\d+,)+)\.mp4\.csmil', webpage, 'video bitrates') | ||||
|         bitrates = [int(b) for b in bitrates.rstrip(',').split(',')] | ||||
|         bitrates.sort() | ||||
|  | ||||
|         formats = [] | ||||
|  | ||||
|         for bitrate in bitrates: | ||||
|             for link in links: | ||||
|                 formats.append({ | ||||
|                     'url': '%s%d.%s' % (link[0], bitrate, link[1]), | ||||
|                     'format_id': '%s-%d' % (link[1], bitrate), | ||||
|                     'vbr': bitrate, | ||||
|                 }) | ||||
|  | ||||
|         post_json = self._search_regex( | ||||
|             r'fb_post\s*=\s*(\{.*?\});', webpage, 'post details') | ||||
|         post = json.loads(post_json) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'ext': 'mp4', | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'thumbnail': thumbnail, | ||||
|             'title': post['name'], | ||||
|             'description': post.get('description'), | ||||
|             'thumbnail': post.get('picture'), | ||||
|             'formats': formats, | ||||
|         } | ||||
|   | ||||
| @@ -15,7 +15,7 @@ class GamekingsIE(InfoExtractor): | ||||
|             'id': '20130811', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Phoenix Wright: Ace Attorney \u2013 Dual Destinies Review', | ||||
|             'description': 'md5:632e61a9f97d700e83f43d77ddafb6a4', | ||||
|             'description': 'md5:36fd701e57e8c15ac8682a2374c99731', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -15,11 +15,12 @@ from ..utils import ( | ||||
| class GameSpotIE(InfoExtractor): | ||||
|     _VALID_URL = r'(?:http://)?(?:www\.)?gamespot\.com/.*-(?P<page_id>\d+)/?' | ||||
|     _TEST = { | ||||
|         "url": "http://www.gamespot.com/arma-iii/videos/arma-iii-community-guide-sitrep-i-6410818/", | ||||
|         "file": "gs-2300-6410818.mp4", | ||||
|         "md5": "b2a30deaa8654fcccd43713a6b6a4825", | ||||
|         "info_dict": { | ||||
|             "title": "Arma 3 - Community Guide: SITREP I", | ||||
|         'url': 'http://www.gamespot.com/videos/arma-3-community-guide-sitrep-i/2300-6410818/', | ||||
|         'md5': 'b2a30deaa8654fcccd43713a6b6a4825', | ||||
|         'info_dict': { | ||||
|             'id': 'gs-2300-6410818', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Arma 3 - Community Guide: SITREP I', | ||||
|             'description': 'Check out this video where some of the basics of Arma 3 is explained.', | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -25,6 +25,7 @@ from ..utils import ( | ||||
| from .brightcove import BrightcoveIE | ||||
| from .ooyala import OoyalaIE | ||||
| from .rutv import RUTVIE | ||||
| from .smotri import SmotriIE | ||||
|  | ||||
|  | ||||
| class GenericIE(InfoExtractor): | ||||
| @@ -34,9 +35,10 @@ class GenericIE(InfoExtractor): | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             'url': 'http://www.hodiho.fr/2013/02/regis-plante-sa-jeep.html', | ||||
|             'file': '13601338388002.mp4', | ||||
|             'md5': '6e15c93721d7ec9e9ca3fdbf07982cfd', | ||||
|             'md5': '85b90ccc9d73b4acd9138d3af4c27f89', | ||||
|             'info_dict': { | ||||
|                 'id': '13601338388002', | ||||
|                 'ext': 'mp4', | ||||
|                 'uploader': 'www.hodiho.fr', | ||||
|                 'title': 'R\u00e9gis plante sa Jeep', | ||||
|             } | ||||
| @@ -45,8 +47,9 @@ class GenericIE(InfoExtractor): | ||||
|         { | ||||
|             'add_ie': ['Bandcamp'], | ||||
|             'url': 'http://bronyrock.com/track/the-pony-mash', | ||||
|             'file': '3235767654.mp3', | ||||
|             'info_dict': { | ||||
|                 'id': '3235767654', | ||||
|                 'ext': 'mp3', | ||||
|                 'title': 'The Pony Mash', | ||||
|                 'uploader': 'M_Pallante', | ||||
|             }, | ||||
| @@ -72,15 +75,27 @@ class GenericIE(InfoExtractor): | ||||
|         { | ||||
|             # https://github.com/rg3/youtube-dl/issues/2253 | ||||
|             'url': 'http://bcove.me/i6nfkrc3', | ||||
|             'file': '3101154703001.mp4', | ||||
|             'md5': '0ba9446db037002366bab3b3eb30c88c', | ||||
|             'info_dict': { | ||||
|                 'id': '3101154703001', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Still no power', | ||||
|                 'uploader': 'thestar.com', | ||||
|                 'description': 'Mississauga resident David Farmer is still out of power as a result of the ice storm a month ago. To keep the house warm, Farmer cuts wood from his property for a wood burning stove downstairs.', | ||||
|             }, | ||||
|             'add_ie': ['Brightcove'], | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://www.championat.com/video/football/v/87/87499.html', | ||||
|             'md5': 'fb973ecf6e4a78a67453647444222983', | ||||
|             'info_dict': { | ||||
|                 'id': '3414141473001', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Видео. Удаление Дзагоева (ЦСКА)', | ||||
|                 'description': 'Онлайн-трансляция матча ЦСКА - "Волга"', | ||||
|                 'uploader': 'Championat', | ||||
|             }, | ||||
|         }, | ||||
|         # Direct link to a video | ||||
|         { | ||||
|             'url': 'http://media.w3.org/2010/05/sintel/trailer.mp4', | ||||
| @@ -102,20 +117,6 @@ class GenericIE(InfoExtractor): | ||||
|                 'title': '2cc213299525360.mov',  # that's what we get | ||||
|             }, | ||||
|         }, | ||||
|         # second style of embedded ooyala videos | ||||
|         { | ||||
|             'url': 'http://www.smh.com.au/tv/business/show/financial-review-sunday/behind-the-scenes-financial-review-sunday--4350201.html', | ||||
|             'info_dict': { | ||||
|                 'id': '13djJjYjptA1XpPx8r9kuzPyj3UZH0Uk', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Behind-the-scenes: Financial Review Sunday ', | ||||
|                 'description': 'Step inside Channel Nine studios for an exclusive tour of its upcoming financial business show.', | ||||
|             }, | ||||
|             'params': { | ||||
|                 # m3u8 download | ||||
|                 'skip_download': True, | ||||
|             }, | ||||
|         }, | ||||
|         # google redirect | ||||
|         { | ||||
|             'url': 'http://www.google.com/url?sa=t&rct=j&q=&esrc=s&source=web&cd=1&cad=rja&ved=0CCUQtwIwAA&url=http%3A%2F%2Fwww.youtube.com%2Fwatch%3Fv%3DcmQHVoWB5FY&ei=F-sNU-LLCaXk4QT52ICQBQ&usg=AFQjCNEw4hL29zgOohLXvpJ-Bdh2bils1Q&bvm=bv.61965928,d.bGE', | ||||
| @@ -186,6 +187,17 @@ class GenericIE(InfoExtractor): | ||||
|                 'description': 'md5:ddb2a40ecd6b6a147e400e535874947b', | ||||
|             } | ||||
|         }, | ||||
|         # Embeded Ustream video | ||||
|         { | ||||
|             'url': 'http://www.american.edu/spa/pti/nsa-privacy-janus-2014.cfm', | ||||
|             'md5': '27b99cdb639c9b12a79bca876a073417', | ||||
|             'info_dict': { | ||||
|                 'id': '45734260', | ||||
|                 'ext': 'flv', | ||||
|                 'uploader': 'AU SPA:  The NSA and Privacy', | ||||
|                 'title': 'NSA and Privacy Forum Debate featuring General Hayden and Barton Gellman' | ||||
|             } | ||||
|         }, | ||||
|         # nowvideo embed hidden behind percent encoding | ||||
|         { | ||||
|             'url': 'http://www.waoanime.tv/the-super-dimension-fortress-macross-episode-1/', | ||||
| @@ -197,6 +209,86 @@ class GenericIE(InfoExtractor): | ||||
|                 'description': 'No description', | ||||
|             }, | ||||
|         }, | ||||
|         # arte embed | ||||
|         { | ||||
|             'url': 'http://www.tv-replay.fr/redirection/20-03-14/x-enius-arte-10753389.html', | ||||
|             'md5': '7653032cbb25bf6c80d80f217055fa43', | ||||
|             'info_dict': { | ||||
|                 'id': '048195-004_PLUS7-F', | ||||
|                 'ext': 'flv', | ||||
|                 'title': 'X:enius', | ||||
|                 'description': 'md5:d5fdf32ef6613cdbfd516ae658abf168', | ||||
|                 'upload_date': '20140320', | ||||
|             }, | ||||
|             'params': { | ||||
|                 'skip_download': 'Requires rtmpdump' | ||||
|             } | ||||
|         }, | ||||
|         # smotri embed | ||||
|         { | ||||
|             'url': 'http://rbctv.rbc.ru/archive/news/562949990879132.shtml', | ||||
|             'md5': 'ec40048448e9284c9a1de77bb188108b', | ||||
|             'info_dict': { | ||||
|                 'id': 'v27008541fad', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Крым и Севастополь вошли в состав России', | ||||
|                 'description': 'md5:fae01b61f68984c7bd2fa741e11c3175', | ||||
|                 'duration': 900, | ||||
|                 'upload_date': '20140318', | ||||
|                 'uploader': 'rbctv_2012_4', | ||||
|                 'uploader_id': 'rbctv_2012_4', | ||||
|             }, | ||||
|         }, | ||||
|         # Condé Nast embed | ||||
|         { | ||||
|             'url': 'http://www.wired.com/2014/04/honda-asimo/', | ||||
|             'md5': 'ba0dfe966fa007657bd1443ee672db0f', | ||||
|             'info_dict': { | ||||
|                 'id': '53501be369702d3275860000', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Honda’s  New Asimo Robot Is More Human Than Ever', | ||||
|             } | ||||
|         }, | ||||
|         # Dailymotion embed | ||||
|         { | ||||
|             'url': 'http://www.spi0n.com/zap-spi0n-com-n216/', | ||||
|             'md5': '441aeeb82eb72c422c7f14ec533999cd', | ||||
|             'info_dict': { | ||||
|                 'id': 'k2mm4bCdJ6CQ2i7c8o2', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Le Zap de Spi0n n°216 - Zapping du Web', | ||||
|                 'uploader': 'Spi0n', | ||||
|             }, | ||||
|             'add_ie': ['Dailymotion'], | ||||
|         }, | ||||
|         # YouTube embed | ||||
|         { | ||||
|             'url': 'http://www.badzine.de/ansicht/datum/2014/06/09/so-funktioniert-die-neue-englische-badminton-liga.html', | ||||
|             'info_dict': { | ||||
|                 'id': 'FXRb4ykk4S0', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'The NBL Auction 2014', | ||||
|                 'uploader': 'BADMINTON England', | ||||
|                 'uploader_id': 'BADMINTONEvents', | ||||
|                 'upload_date': '20140603', | ||||
|                 'description': 'md5:9ef128a69f1e262a700ed83edb163a73', | ||||
|             }, | ||||
|             'add_ie': ['Youtube'], | ||||
|             'params': { | ||||
|                 'skip_download': True, | ||||
|             } | ||||
|         }, | ||||
|         # MTVSercices embed | ||||
|         { | ||||
|             'url': 'http://www.gametrailers.com/news-post/76093/north-america-europe-is-getting-that-mario-kart-8-mercedes-dlc-too', | ||||
|             'md5': '35727f82f58c76d996fc188f9755b0d5', | ||||
|             'info_dict': { | ||||
|                 'id': '0306a69b-8adf-4fb5-aace-75f8e8cbfca9', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Review', | ||||
|                 'description': 'Mario\'s life in the fast lane has never looked so good.', | ||||
|             }, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     def report_download_webpage(self, video_id): | ||||
| @@ -281,17 +373,31 @@ class GenericIE(InfoExtractor): | ||||
|         } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         if url.startswith('//'): | ||||
|             return { | ||||
|                 '_type': 'url', | ||||
|                 'url': self.http_scheme() + url, | ||||
|             } | ||||
|  | ||||
|         parsed_url = compat_urlparse.urlparse(url) | ||||
|         if not parsed_url.scheme: | ||||
|             default_search = self._downloader.params.get('default_search') | ||||
|             if default_search is None: | ||||
|                 default_search = 'auto' | ||||
|                 default_search = 'auto_warning' | ||||
|  | ||||
|             if default_search == 'auto': | ||||
|             if default_search in ('auto', 'auto_warning'): | ||||
|                 if '/' in url: | ||||
|                     self._downloader.report_warning('The url doesn\'t specify the protocol, trying with http') | ||||
|                     return self.url_result('http://' + url) | ||||
|                 else: | ||||
|                     if default_search == 'auto_warning': | ||||
|                         if re.match(r'^(?:url|URL)$', url): | ||||
|                             raise ExtractorError( | ||||
|                                 'Invalid URL:  %r . Call youtube-dl like this:  youtube-dl -v "https://www.youtube.com/watch?v=BaW_jenozKc"  ' % url, | ||||
|                                 expected=True) | ||||
|                         else: | ||||
|                             self._downloader.report_warning( | ||||
|                                 'Falling back to youtube search for  %s . Set --default-search to "auto" to suppress this warning.' % url) | ||||
|                     return self.url_result('ytsearch:' + url) | ||||
|             else: | ||||
|                 assert ':' in default_search | ||||
| @@ -400,8 +506,13 @@ class GenericIE(InfoExtractor): | ||||
|  | ||||
|         # Look for embedded YouTube player | ||||
|         matches = re.findall(r'''(?x) | ||||
|             (?:<iframe[^>]+?src=|embedSWF\(\s*) | ||||
|             (["\'])(?P<url>(?:https?:)?//(?:www\.)?youtube\.com/ | ||||
|             (?: | ||||
|                 <iframe[^>]+?src=| | ||||
|                 <embed[^>]+?src=| | ||||
|                 embedSWF\(?:\s* | ||||
|             ) | ||||
|             (["\']) | ||||
|                 (?P<url>(?:https?:)?//(?:www\.)?youtube\.com/ | ||||
|                 (?:embed|v)/.+?) | ||||
|             \1''', webpage) | ||||
|         if matches: | ||||
| @@ -414,7 +525,7 @@ class GenericIE(InfoExtractor): | ||||
|         matches = re.findall( | ||||
|             r'<iframe[^>]+?src=(["\'])(?P<url>(?:https?:)?//(?:www\.)?dailymotion\.com/embed/video/.+?)\1', webpage) | ||||
|         if matches: | ||||
|             urlrs = [self.url_result(unescapeHTML(tuppl[1]), 'Dailymotion') | ||||
|             urlrs = [self.url_result(unescapeHTML(tuppl[1])) | ||||
|                      for tuppl in matches] | ||||
|             return self.playlist_result( | ||||
|                 urlrs, playlist_id=video_id, playlist_title=video_title) | ||||
| @@ -440,6 +551,22 @@ class GenericIE(InfoExtractor): | ||||
|         if mobj: | ||||
|             return self.url_result(mobj.group(1), 'BlipTV') | ||||
|  | ||||
|         # Look for embedded condenast player | ||||
|         matches = re.findall( | ||||
|             r'<iframe\s+(?:[a-zA-Z-]+="[^"]+"\s+)*?src="(https?://player\.cnevids\.com/embed/[^"]+")', | ||||
|             webpage) | ||||
|         if matches: | ||||
|             return { | ||||
|                 '_type': 'playlist', | ||||
|                 'entries': [{ | ||||
|                     '_type': 'url', | ||||
|                     'ie_key': 'CondeNast', | ||||
|                     'url': ma, | ||||
|                 } for ma in matches], | ||||
|                 'title': video_title, | ||||
|                 'id': video_id, | ||||
|             } | ||||
|  | ||||
|         # Look for Bandcamp pages with custom domain | ||||
|         mobj = re.search(r'<meta property="og:url"[^>]*?content="(.*?bandcamp\.com.*?)"', webpage) | ||||
|         if mobj is not None: | ||||
| @@ -460,7 +587,7 @@ class GenericIE(InfoExtractor): | ||||
|             return OoyalaIE._build_url_result(mobj.group('ec')) | ||||
|  | ||||
|         # Look for Aparat videos | ||||
|         mobj = re.search(r'<iframe src="(http://www\.aparat\.com/video/[^"]+)"', webpage) | ||||
|         mobj = re.search(r'<iframe .*?src="(http://www\.aparat\.com/video/[^"]+)"', webpage) | ||||
|         if mobj is not None: | ||||
|             return self.url_result(mobj.group(1), 'Aparat') | ||||
|  | ||||
| @@ -469,17 +596,18 @@ class GenericIE(InfoExtractor): | ||||
|         if mobj is not None: | ||||
|             return self.url_result(mobj.group(1), 'Mpora') | ||||
|  | ||||
|         # Look for embedded NovaMov player | ||||
|         # Look for embedded NovaMov-based player | ||||
|         mobj = re.search( | ||||
|             r'<iframe[^>]+?src=(["\'])(?P<url>http://(?:(?:embed|www)\.)?novamov\.com/embed\.php.+?)\1', webpage) | ||||
|             r'''(?x)<(?:pagespeed_)?iframe[^>]+?src=(["\']) | ||||
|                     (?P<url>http://(?:(?:embed|www)\.)? | ||||
|                         (?:novamov\.com| | ||||
|                            nowvideo\.(?:ch|sx|eu|at|ag|co)| | ||||
|                            videoweed\.(?:es|com)| | ||||
|                            movshare\.(?:net|sx|ag)| | ||||
|                            divxstage\.(?:eu|net|ch|co|at|ag)) | ||||
|                         /embed\.php.+?)\1''', webpage) | ||||
|         if mobj is not None: | ||||
|             return self.url_result(mobj.group('url'), 'NovaMov') | ||||
|  | ||||
|         # Look for embedded NowVideo player | ||||
|         mobj = re.search( | ||||
|             r'<iframe[^>]+?src=(["\'])(?P<url>http://(?:(?:embed|www)\.)?nowvideo\.(?:ch|sx|eu)/embed\.php.+?)\1', webpage) | ||||
|         if mobj is not None: | ||||
|             return self.url_result(mobj.group('url'), 'NowVideo') | ||||
|             return self.url_result(mobj.group('url')) | ||||
|  | ||||
|         # Look for embedded Facebook player | ||||
|         mobj = re.search( | ||||
| @@ -525,65 +653,120 @@ class GenericIE(InfoExtractor): | ||||
|         if mobj is not None: | ||||
|             return self.url_result(mobj.group('url'), 'TED') | ||||
|  | ||||
|         # Start with something easy: JW Player in SWFObject | ||||
|         mobj = re.search(r'flashvars: [\'"](?:.*&)?file=(http[^\'"&]*)', webpage) | ||||
|         if mobj is None: | ||||
|             # Look for gorilla-vid style embedding | ||||
|             mobj = re.search(r'(?s)(?:jw_plugins|JWPlayerOptions).*?file\s*:\s*["\'](.*?)["\']', webpage) | ||||
|         if mobj is None: | ||||
|             # Broaden the search a little bit | ||||
|             mobj = re.search(r'[^A-Za-z0-9]?(?:file|source)=(http[^\'"&]*)', webpage) | ||||
|         if mobj is None: | ||||
|             # Broaden the search a little bit: JWPlayer JS loader | ||||
|             mobj = re.search(r'[^A-Za-z0-9]?file["\']?:\s*["\'](http(?![^\'"]+\.[0-9]+[\'"])[^\'"]+)["\']', webpage) | ||||
|         # Look for embedded Ustream videos | ||||
|         mobj = re.search( | ||||
|             r'<iframe[^>]+?src=(["\'])(?P<url>http://www\.ustream\.tv/embed/.+?)\1', webpage) | ||||
|         if mobj is not None: | ||||
|             return self.url_result(mobj.group('url'), 'Ustream') | ||||
|  | ||||
|         if mobj is None: | ||||
|         # Look for embedded arte.tv player | ||||
|         mobj = re.search( | ||||
|             r'<script [^>]*?src="(?P<url>http://www\.arte\.tv/playerv2/embed[^"]+)"', | ||||
|             webpage) | ||||
|         if mobj is not None: | ||||
|             return self.url_result(mobj.group('url'), 'ArteTVEmbed') | ||||
|  | ||||
|         # Look for embedded smotri.com player | ||||
|         smotri_url = SmotriIE._extract_url(webpage) | ||||
|         if smotri_url: | ||||
|             return self.url_result(smotri_url, 'Smotri') | ||||
|  | ||||
|         # Look for embeded soundcloud player | ||||
|         mobj = re.search( | ||||
|             r'<iframe src="(?P<url>https?://(?:w\.)?soundcloud\.com/player[^"]+)"', | ||||
|             webpage) | ||||
|         if mobj is not None: | ||||
|             url = unescapeHTML(mobj.group('url')) | ||||
|             return self.url_result(url) | ||||
|  | ||||
|         # Look for embedded vulture.com player | ||||
|         mobj = re.search( | ||||
|             r'<iframe src="(?P<url>https?://video\.vulture\.com/[^"]+)"', | ||||
|             webpage) | ||||
|         if mobj is not None: | ||||
|             url = unescapeHTML(mobj.group('url')) | ||||
|             return self.url_result(url, ie='Vulture') | ||||
|  | ||||
|         # Look for embedded mtvservices player | ||||
|         mobj = re.search( | ||||
|             r'<iframe src="(?P<url>https?://media\.mtvnservices\.com/embed/[^"]+)"', | ||||
|             webpage) | ||||
|         if mobj is not None: | ||||
|             url = unescapeHTML(mobj.group('url')) | ||||
|             return self.url_result(url, ie='MTVServicesEmbedded') | ||||
|  | ||||
|         # Start with something easy: JW Player in SWFObject | ||||
|         found = re.findall(r'flashvars: [\'"](?:.*&)?file=(http[^\'"&]*)', webpage) | ||||
|         if not found: | ||||
|             # Look for gorilla-vid style embedding | ||||
|             found = re.findall(r'''(?sx) | ||||
|                 (?: | ||||
|                     jw_plugins| | ||||
|                     JWPlayerOptions| | ||||
|                     jwplayer\s*\(\s*["'][^'"]+["']\s*\)\s*\.setup | ||||
|                 ) | ||||
|                 .*?file\s*:\s*["\'](.*?)["\']''', webpage) | ||||
|         if not found: | ||||
|             # Broaden the search a little bit | ||||
|             found = re.findall(r'[^A-Za-z0-9]?(?:file|source)=(http[^\'"&]*)', webpage) | ||||
|         if not found: | ||||
|             # Broaden the findall a little bit: JWPlayer JS loader | ||||
|             found = re.findall(r'[^A-Za-z0-9]?file["\']?:\s*["\'](http(?![^\'"]+\.[0-9]+[\'"])[^\'"]+)["\']', webpage) | ||||
|         if not found: | ||||
|             # Try to find twitter cards info | ||||
|             mobj = re.search(r'<meta (?:property|name)="twitter:player:stream" (?:content|value)="(.+?)"', webpage) | ||||
|         if mobj is None: | ||||
|             found = re.findall(r'<meta (?:property|name)="twitter:player:stream" (?:content|value)="(.+?)"', webpage) | ||||
|         if not found: | ||||
|             # We look for Open Graph info: | ||||
|             # We have to match any number spaces between elements, some sites try to align them (eg.: statigr.am) | ||||
|             m_video_type = re.search(r'<meta.*?property="og:video:type".*?content="video/(.*?)"', webpage) | ||||
|             m_video_type = re.findall(r'<meta.*?property="og:video:type".*?content="video/(.*?)"', webpage) | ||||
|             # We only look in og:video if the MIME type is a video, don't try if it's a Flash player: | ||||
|             if m_video_type is not None: | ||||
|                 mobj = re.search(r'<meta.*?property="og:video".*?content="(.*?)"', webpage) | ||||
|         if mobj is None: | ||||
|                 found = re.findall(r'<meta.*?property="og:video".*?content="(.*?)"', webpage) | ||||
|         if not found: | ||||
|             # HTML5 video | ||||
|             mobj = re.search(r'<video[^<]*(?:>.*?<source.*?)? src="([^"]+)"', webpage, flags=re.DOTALL) | ||||
|         if mobj is None: | ||||
|             mobj = re.search( | ||||
|             found = re.findall(r'(?s)<video[^<]*(?:>.*?<source.*?)? src="([^"]+)"', webpage) | ||||
|         if not found: | ||||
|             found = re.search( | ||||
|                 r'(?i)<meta\s+(?=(?:[a-z-]+="[^"]+"\s+)*http-equiv="refresh")' | ||||
|                 r'(?:[a-z-]+="[^"]+"\s+)*?content="[0-9]{,2};url=\'([^\']+)\'"', | ||||
|                 webpage) | ||||
|             if mobj: | ||||
|                 new_url = mobj.group(1) | ||||
|             if found: | ||||
|                 new_url = found.group(1) | ||||
|                 self.report_following_redirect(new_url) | ||||
|                 return { | ||||
|                     '_type': 'url', | ||||
|                     'url': new_url, | ||||
|                 } | ||||
|         if mobj is None: | ||||
|         if not found: | ||||
|             raise ExtractorError('Unsupported URL: %s' % url) | ||||
|  | ||||
|         # It's possible that one of the regexes | ||||
|         # matched, but returned an empty group: | ||||
|         if mobj.group(1) is None: | ||||
|             raise ExtractorError('Did not find a valid video URL at %s' % url) | ||||
|         entries = [] | ||||
|         for video_url in found: | ||||
|             video_url = compat_urlparse.urljoin(url, video_url) | ||||
|             video_id = compat_urllib_parse.unquote(os.path.basename(video_url)) | ||||
|  | ||||
|         video_url = mobj.group(1) | ||||
|         video_url = compat_urlparse.urljoin(url, video_url) | ||||
|         video_id = compat_urllib_parse.unquote(os.path.basename(video_url)) | ||||
|             # Sometimes, jwplayer extraction will result in a YouTube URL | ||||
|             if YoutubeIE.suitable(video_url): | ||||
|                 entries.append(self.url_result(video_url, 'Youtube')) | ||||
|                 continue | ||||
|  | ||||
|         # Sometimes, jwplayer extraction will result in a YouTube URL | ||||
|         if YoutubeIE.suitable(video_url): | ||||
|             return self.url_result(video_url, 'Youtube') | ||||
|             # here's a fun little line of code for you: | ||||
|             video_id = os.path.splitext(video_id)[0] | ||||
|  | ||||
|         # here's a fun little line of code for you: | ||||
|         video_id = os.path.splitext(video_id)[0] | ||||
|             entries.append({ | ||||
|                 'id': video_id, | ||||
|                 'url': video_url, | ||||
|                 'uploader': video_uploader, | ||||
|                 'title': video_title, | ||||
|             }) | ||||
|  | ||||
|         if len(entries) == 1: | ||||
|             return entries[0] | ||||
|         else: | ||||
|             for num, e in enumerate(entries, start=1): | ||||
|                 e['title'] = '%s (%d)' % (e['title'], num) | ||||
|             return { | ||||
|                 '_type': 'playlist', | ||||
|                 'entries': entries, | ||||
|             } | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'uploader': video_uploader, | ||||
|             'title': video_title, | ||||
|         } | ||||
|   | ||||
							
								
								
									
										76
									
								
								youtube_dl/extractor/gorillavid.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								youtube_dl/extractor/gorillavid.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     determine_ext, | ||||
|     compat_urllib_parse, | ||||
|     compat_urllib_request, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class GorillaVidIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?gorillavid\.in/(?:embed-)?(?P<id>[0-9a-zA-Z]+)(?:-[0-9]+x[0-9]+\.html)?' | ||||
|  | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://gorillavid.in/06y9juieqpmi', | ||||
|         'md5': '5ae4a3580620380619678ee4875893ba', | ||||
|         'info_dict': { | ||||
|             'id': '06y9juieqpmi', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Rebecca Black My Moment Official Music Video Reaction', | ||||
|             'thumbnail': 're:http://.*\.jpg', | ||||
|         }, | ||||
|     }, { | ||||
|         'url': 'http://gorillavid.in/embed-z08zf8le23c6-960x480.html', | ||||
|         'md5': 'c9e293ca74d46cad638e199c3f3fe604', | ||||
|         'info_dict': { | ||||
|             'id': 'z08zf8le23c6', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Say something nice', | ||||
|             'thumbnail': 're:http://.*\.jpg', | ||||
|         }, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         url = 'http://gorillavid.in/%s' % video_id | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         fields = dict(re.findall(r'''(?x)<input\s+ | ||||
|             type="hidden"\s+ | ||||
|             name="([^"]+)"\s+ | ||||
|             (?:id="[^"]+"\s+)? | ||||
|             value="([^"]*)" | ||||
|             ''', webpage)) | ||||
|          | ||||
|         if fields['op'] == 'download1': | ||||
|             post = compat_urllib_parse.urlencode(fields) | ||||
|  | ||||
|             req = compat_urllib_request.Request(url, post) | ||||
|             req.add_header('Content-type', 'application/x-www-form-urlencoded') | ||||
|  | ||||
|             webpage = self._download_webpage(req, video_id, 'Downloading video page') | ||||
|  | ||||
|         title = self._search_regex(r'style="z-index: [0-9]+;">([0-9a-zA-Z ]+)(?:-.+)?</span>', webpage, 'title') | ||||
|         thumbnail = self._search_regex(r'image:\'(http[^\']+)\',', webpage, 'thumbnail') | ||||
|         url = self._search_regex(r'file: \'(http[^\']+)\',', webpage, 'file url') | ||||
|  | ||||
|         formats = [{ | ||||
|             'format_id': 'sd', | ||||
|             'url': url, | ||||
|             'ext': determine_ext(url), | ||||
|             'quality': 1, | ||||
|         }] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'formats': formats, | ||||
|         } | ||||
							
								
								
									
										42
									
								
								youtube_dl/extractor/hentaistigma.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								youtube_dl/extractor/hentaistigma.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class HentaiStigmaIE(InfoExtractor): | ||||
|     _VALID_URL = r'^https?://hentai\.animestigma\.com/(?P<id>[^/]+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://hentai.animestigma.com/inyouchuu-etsu-bonus/', | ||||
|         'md5': '4e3d07422a68a4cc363d8f57c8bf0d23', | ||||
|         'info_dict': { | ||||
|             'id': 'inyouchuu-etsu-bonus', | ||||
|             'ext': 'mp4', | ||||
|             "title": "Inyouchuu Etsu Bonus", | ||||
|             "age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         title = self._html_search_regex( | ||||
|             r'<h2 class="posttitle"><a[^>]*>([^<]+)</a>', | ||||
|             webpage, 'title') | ||||
|         wrap_url = self._html_search_regex( | ||||
|             r'<iframe src="([^"]+mp4)"', webpage, 'wrapper url') | ||||
|         wrap_webpage = self._download_webpage(wrap_url, video_id) | ||||
|  | ||||
|         video_url = self._html_search_regex( | ||||
|             r'clip:\s*{\s*url: "([^"]*)"', wrap_webpage, 'video url') | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'title': title, | ||||
|             'age_limit': 18, | ||||
|         } | ||||
| @@ -21,9 +21,10 @@ class HuffPostIE(InfoExtractor): | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://live.huffingtonpost.com/r/segment/legalese-it/52dd3e4b02a7602131000677', | ||||
|         'file': '52dd3e4b02a7602131000677.mp4', | ||||
|         'md5': '55f5e8981c1c80a64706a44b74833de8', | ||||
|         'info_dict': { | ||||
|             'id': '52dd3e4b02a7602131000677', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Legalese It! with @MikeSacksHP', | ||||
|             'description': 'This week on Legalese It, Mike talks to David Bosco about his new book on the ICC, "Rough Justice," he also discusses the Virginia AG\'s historic stance on gay marriage, the execution of Edgar Tamayo, the ICC\'s delay of Kenya\'s President and more.  ', | ||||
|             'duration': 1549, | ||||
|   | ||||
| @@ -1,10 +1,11 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import json | ||||
| import re | ||||
| import time | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_str, | ||||
|     compat_urllib_parse, | ||||
|     compat_urllib_request, | ||||
|  | ||||
| @@ -13,59 +14,55 @@ from ..utils import ( | ||||
|  | ||||
|  | ||||
| class HypemIE(InfoExtractor): | ||||
|     """Information Extractor for hypem""" | ||||
|     _VALID_URL = r'(?:http://)?(?:www\.)?hypem\.com/track/([^/]+)/([^/]+)' | ||||
|     _VALID_URL = r'http://(?:www\.)?hypem\.com/track/([^/]+)/([^/]+)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://hypem.com/track/1v6ga/BODYWORK+-+TAME', | ||||
|         u'file': u'1v6ga.mp3', | ||||
|         u'md5': u'b9cc91b5af8995e9f0c1cee04c575828', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Tame" | ||||
|         'url': 'http://hypem.com/track/1v6ga/BODYWORK+-+TAME', | ||||
|         'md5': 'b9cc91b5af8995e9f0c1cee04c575828', | ||||
|         'info_dict': { | ||||
|             'id': '1v6ga', | ||||
|             'ext': 'mp3', | ||||
|             'title': 'Tame', | ||||
|             'uploader': 'BODYWORK', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         if mobj is None: | ||||
|             raise ExtractorError(u'Invalid URL: %s' % url) | ||||
|         track_id = mobj.group(1) | ||||
|  | ||||
|         data = {'ax': 1, 'ts': time.time()} | ||||
|         data_encoded = compat_urllib_parse.urlencode(data) | ||||
|         complete_url = url + "?" + data_encoded | ||||
|         request = compat_urllib_request.Request(complete_url) | ||||
|         response, urlh = self._download_webpage_handle(request, track_id, u'Downloading webpage with the url') | ||||
|         response, urlh = self._download_webpage_handle( | ||||
|             request, track_id, 'Downloading webpage with the url') | ||||
|         cookie = urlh.headers.get('Set-Cookie', '') | ||||
|  | ||||
|         self.report_extraction(track_id) | ||||
|  | ||||
|         html_tracks = self._html_search_regex(r'<script type="application/json" id="displayList-data">(.*?)</script>', | ||||
|             response, u'tracks', flags=re.MULTILINE|re.DOTALL).strip() | ||||
|         html_tracks = self._html_search_regex( | ||||
|             r'(?ms)<script type="application/json" id="displayList-data">\s*(.*?)\s*</script>', | ||||
|             response, 'tracks') | ||||
|         try: | ||||
|             track_list = json.loads(html_tracks) | ||||
|             track = track_list[u'tracks'][0] | ||||
|             track = track_list['tracks'][0] | ||||
|         except ValueError: | ||||
|             raise ExtractorError(u'Hypemachine contained invalid JSON.') | ||||
|             raise ExtractorError('Hypemachine contained invalid JSON.') | ||||
|  | ||||
|         key = track[u"key"] | ||||
|         track_id = track[u"id"] | ||||
|         artist = track[u"artist"] | ||||
|         title = track[u"song"] | ||||
|         key = track['key'] | ||||
|         track_id = track['id'] | ||||
|         artist = track['artist'] | ||||
|         title = track['song'] | ||||
|  | ||||
|         serve_url = "http://hypem.com/serve/source/%s/%s" % (compat_str(track_id), compat_str(key)) | ||||
|         request = compat_urllib_request.Request(serve_url, "" , {'Content-Type': 'application/json'}) | ||||
|         serve_url = "http://hypem.com/serve/source/%s/%s" % (track_id, key) | ||||
|         request = compat_urllib_request.Request( | ||||
|             serve_url, '', {'Content-Type': 'application/json'}) | ||||
|         request.add_header('cookie', cookie) | ||||
|         song_data_json = self._download_webpage(request, track_id, u'Downloading metadata') | ||||
|         try: | ||||
|             song_data = json.loads(song_data_json) | ||||
|         except ValueError: | ||||
|             raise ExtractorError(u'Hypemachine contained invalid JSON.') | ||||
|         final_url = song_data[u"url"] | ||||
|         song_data = self._download_json(request, track_id, 'Downloading metadata') | ||||
|         final_url = song_data["url"] | ||||
|  | ||||
|         return [{ | ||||
|             'id':       track_id, | ||||
|             'url':      final_url, | ||||
|             'ext':      "mp3", | ||||
|             'title':    title, | ||||
|             'artist':   artist, | ||||
|         }] | ||||
|         return { | ||||
|             'id': track_id, | ||||
|             'url': final_url, | ||||
|             'ext': 'mp3', | ||||
|             'title': title, | ||||
|             'uploader': artist, | ||||
|         } | ||||
|   | ||||
| @@ -5,8 +5,8 @@ import re | ||||
| from .common import InfoExtractor | ||||
| 
 | ||||
| 
 | ||||
| class StatigramIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(www\.)?statigr\.am/p/(?P<id>[^/]+)' | ||||
| class IconosquareIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(www\.)?(?:iconosquare\.com|statigr\.am)/p/(?P<id>[^/]+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://statigr.am/p/522207370455279102_24101272', | ||||
|         'md5': '6eb93b882a3ded7c378ee1d6884b1814', | ||||
| @@ -15,6 +15,7 @@ class StatigramIE(InfoExtractor): | ||||
|             'ext': 'mp4', | ||||
|             'uploader_id': 'aguynamedpatrick', | ||||
|             'title': 'Instagram photo by @aguynamedpatrick (Patrick Janelle)', | ||||
|             'description': 'md5:644406a9ec27457ed7aa7a9ebcd4ce3d', | ||||
|         }, | ||||
|     } | ||||
| 
 | ||||
| @@ -25,7 +26,7 @@ class StatigramIE(InfoExtractor): | ||||
|         html_title = self._html_search_regex( | ||||
|             r'<title>(.+?)</title>', | ||||
|             webpage, 'title') | ||||
|         title = re.sub(r'(?: *\(Videos?\))? \| Statigram$', '', html_title) | ||||
|         title = re.sub(r'(?: *\(Videos?\))? \| (?:Iconosquare|Statigram)$', '', html_title) | ||||
|         uploader_id = self._html_search_regex( | ||||
|             r'@([^ ]+)', title, 'uploader name', fatal=False) | ||||
| 
 | ||||
| @@ -33,6 +34,7 @@ class StatigramIE(InfoExtractor): | ||||
|             'id': video_id, | ||||
|             'url': self._og_search_video_url(webpage), | ||||
|             'title': title, | ||||
|             'description': self._og_search_description(webpage), | ||||
|             'thumbnail': self._og_search_thumbnail(webpage), | ||||
|             'uploader_id': uploader_id | ||||
|         } | ||||
| @@ -1,10 +1,8 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import json | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     determine_ext, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class IGNIE(InfoExtractor): | ||||
| @@ -14,52 +12,57 @@ class IGNIE(InfoExtractor): | ||||
|     """ | ||||
|  | ||||
|     _VALID_URL = r'https?://.+?\.ign\.com/(?P<type>videos|show_videos|articles|(?:[^/]*/feature))(/.+)?/(?P<name_or_id>.+)' | ||||
|     IE_NAME = u'ign.com' | ||||
|     IE_NAME = 'ign.com' | ||||
|  | ||||
|     _CONFIG_URL_TEMPLATE = 'http://www.ign.com/videos/configs/id/%s.config' | ||||
|     _DESCRIPTION_RE = [r'<span class="page-object-description">(.+?)</span>', | ||||
|                        r'id="my_show_video">.*?<p>(.*?)</p>', | ||||
|                        ] | ||||
|     _DESCRIPTION_RE = [ | ||||
|         r'<span class="page-object-description">(.+?)</span>', | ||||
|         r'id="my_show_video">.*?<p>(.*?)</p>', | ||||
|     ] | ||||
|  | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             u'url': u'http://www.ign.com/videos/2013/06/05/the-last-of-us-review', | ||||
|             u'file': u'8f862beef863986b2785559b9e1aa599.mp4', | ||||
|             u'md5': u'eac8bdc1890980122c3b66f14bdd02e9', | ||||
|             u'info_dict': { | ||||
|                 u'title': u'The Last of Us Review', | ||||
|                 u'description': u'md5:c8946d4260a4d43a00d5ae8ed998870c', | ||||
|             'url': 'http://www.ign.com/videos/2013/06/05/the-last-of-us-review', | ||||
|             'md5': 'eac8bdc1890980122c3b66f14bdd02e9', | ||||
|             'info_dict': { | ||||
|                 'id': '8f862beef863986b2785559b9e1aa599', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'The Last of Us Review', | ||||
|                 'description': 'md5:c8946d4260a4d43a00d5ae8ed998870c', | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             u'url': u'http://me.ign.com/en/feature/15775/100-little-things-in-gta-5-that-will-blow-your-mind', | ||||
|             u'playlist': [ | ||||
|             'url': 'http://me.ign.com/en/feature/15775/100-little-things-in-gta-5-that-will-blow-your-mind', | ||||
|             'playlist': [ | ||||
|                 { | ||||
|                     u'file': u'5ebbd138523268b93c9141af17bec937.mp4', | ||||
|                     u'info_dict': { | ||||
|                         u'title': u'GTA 5 Video Review', | ||||
|                         u'description': u'Rockstar drops the mic on this generation of games. Watch our review of the masterly Grand Theft Auto V.', | ||||
|                     'info_dict': { | ||||
|                         'id': '5ebbd138523268b93c9141af17bec937', | ||||
|                         'ext': 'mp4', | ||||
|                         'title': 'GTA 5 Video Review', | ||||
|                         'description': 'Rockstar drops the mic on this generation of games. Watch our review of the masterly Grand Theft Auto V.', | ||||
|                     }, | ||||
|                 }, | ||||
|                 { | ||||
|                     u'file': u'638672ee848ae4ff108df2a296418ee2.mp4', | ||||
|                     u'info_dict': { | ||||
|                         u'title': u'26 Twisted Moments from GTA 5 in Slow Motion', | ||||
|                         u'description': u'The twisted beauty of GTA 5 in stunning slow motion.', | ||||
|                     'info_dict': { | ||||
|                         'id': '638672ee848ae4ff108df2a296418ee2', | ||||
|                         'ext': 'mp4', | ||||
|                         'title': '26 Twisted Moments from GTA 5 in Slow Motion', | ||||
|                         'description': 'The twisted beauty of GTA 5 in stunning slow motion.', | ||||
|                     }, | ||||
|                 }, | ||||
|             ], | ||||
|             u'params': { | ||||
|                 u'skip_download': True, | ||||
|             'params': { | ||||
|                 'skip_download': True, | ||||
|             }, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     def _find_video_id(self, webpage): | ||||
|         res_id = [r'data-video-id="(.+?)"', | ||||
|                   r'<object id="vid_(.+?)"', | ||||
|                   r'<meta name="og:image" content=".*/(.+?)-(.+?)/.+.jpg"', | ||||
|                   ] | ||||
|         res_id = [ | ||||
|             r'data-video-id="(.+?)"', | ||||
|             r'<object id="vid_(.+?)"', | ||||
|             r'<meta name="og:image" content=".*/(.+?)-(.+?)/.+.jpg"', | ||||
|         ] | ||||
|         return self._search_regex(res_id, webpage, 'video id') | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
| @@ -68,7 +71,7 @@ class IGNIE(InfoExtractor): | ||||
|         page_type = mobj.group('type') | ||||
|         webpage = self._download_webpage(url, name_or_id) | ||||
|         if page_type == 'articles': | ||||
|             video_url = self._search_regex(r'var videoUrl = "(.+?)"', webpage, u'video url') | ||||
|             video_url = self._search_regex(r'var videoUrl = "(.+?)"', webpage, 'video url') | ||||
|             return self.url_result(video_url, ie='IGN') | ||||
|         elif page_type != 'video': | ||||
|             multiple_urls = re.findall( | ||||
| @@ -80,50 +83,42 @@ class IGNIE(InfoExtractor): | ||||
|         video_id = self._find_video_id(webpage) | ||||
|         result = self._get_video_info(video_id) | ||||
|         description = self._html_search_regex(self._DESCRIPTION_RE, | ||||
|                                               webpage, 'video description', | ||||
|                                               flags=re.DOTALL) | ||||
|             webpage, 'video description', flags=re.DOTALL) | ||||
|         result['description'] = description | ||||
|         return result | ||||
|  | ||||
|     def _get_video_info(self, video_id): | ||||
|         config_url = self._CONFIG_URL_TEMPLATE % video_id | ||||
|         config = json.loads(self._download_webpage(config_url, video_id, | ||||
|                             u'Downloading video info')) | ||||
|         config = self._download_json(config_url, video_id) | ||||
|         media = config['playlist']['media'] | ||||
|         video_url = media['url'] | ||||
|  | ||||
|         return {'id': media['metadata']['videoId'], | ||||
|                 'url': video_url, | ||||
|                 'ext': determine_ext(video_url), | ||||
|                 'title': media['metadata']['title'], | ||||
|                 'thumbnail': media['poster'][0]['url'].replace('{size}', 'grande'), | ||||
|                 } | ||||
|         return { | ||||
|             'id': media['metadata']['videoId'], | ||||
|             'url': media['url'], | ||||
|             'title': media['metadata']['title'], | ||||
|             'thumbnail': media['poster'][0]['url'].replace('{size}', 'grande'), | ||||
|         } | ||||
|  | ||||
|  | ||||
| class OneUPIE(IGNIE): | ||||
|     """Extractor for 1up.com, it uses the ign videos system.""" | ||||
|  | ||||
|     _VALID_URL = r'https?://gamevideos\.1up\.com/(?P<type>video)/id/(?P<name_or_id>.+)' | ||||
|     IE_NAME = '1up.com' | ||||
|  | ||||
|     _DESCRIPTION_RE = r'<div id="vid_summary">(.+?)</div>' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'url': u'http://gamevideos.1up.com/video/id/34976', | ||||
|         u'file': u'34976.mp4', | ||||
|         u'md5': u'68a54ce4ebc772e4b71e3123d413163d', | ||||
|         u'info_dict': { | ||||
|             u'title': u'Sniper Elite V2 - Trailer', | ||||
|             u'description': u'md5:5d289b722f5a6d940ca3136e9dae89cf', | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://gamevideos.1up.com/video/id/34976', | ||||
|         'md5': '68a54ce4ebc772e4b71e3123d413163d', | ||||
|         'info_dict': { | ||||
|             'id': '34976', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Sniper Elite V2 - Trailer', | ||||
|             'description': 'md5:5d289b722f5a6d940ca3136e9dae89cf', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     # Override IGN tests | ||||
|     _TESTS = [] | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         id = mobj.group('name_or_id') | ||||
|         result = super(OneUPIE, self)._real_extract(url) | ||||
|         result['id'] = id | ||||
|         result['id'] = mobj.group('name_or_id') | ||||
|         return result | ||||
|   | ||||
| @@ -11,16 +11,15 @@ from ..utils import ( | ||||
|  | ||||
| class InfoQIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?infoq\.com/[^/]+/(?P<id>[^/]+)$' | ||||
|  | ||||
|     _TEST = { | ||||
|         "name": "InfoQ", | ||||
|         "url": "http://www.infoq.com/presentations/A-Few-of-My-Favorite-Python-Things", | ||||
|         "file": "12-jan-pythonthings.mp4", | ||||
|         "info_dict": { | ||||
|             "description": "Mike Pirnat presents some tips and tricks, standard libraries and third party packages that make programming in Python a richer experience.", | ||||
|             "title": "A Few of My Favorite [Python] Things", | ||||
|         }, | ||||
|         "params": { | ||||
|             "skip_download": True, | ||||
|         'url': 'http://www.infoq.com/presentations/A-Few-of-My-Favorite-Python-Things', | ||||
|         'md5': 'b5ca0e0a8c1fed93b0e65e48e462f9a2', | ||||
|         'info_dict': { | ||||
|             'id': '12-jan-pythonthings', | ||||
|             'ext': 'mp4', | ||||
|             'description': 'Mike Pirnat presents some tips and tricks, standard libraries and third party packages that make programming in Python a richer experience.', | ||||
|             'title': 'A Few of My Favorite [Python] Things', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
| @@ -30,26 +29,39 @@ class InfoQIE(InfoExtractor): | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<title>(.*?)</title>', webpage, 'title') | ||||
|         video_description = self._html_search_meta('description', webpage, 'description') | ||||
|  | ||||
|         # The server URL is hardcoded | ||||
|         video_url = 'rtmpe://video.infoq.com/cfx/st/' | ||||
|  | ||||
|         # Extract video URL | ||||
|         encoded_id = self._search_regex(r"jsclassref ?= ?'([^']*)'", webpage, 'encoded id') | ||||
|         encoded_id = self._search_regex( | ||||
|             r"jsclassref\s*=\s*'([^']*)'", webpage, 'encoded id') | ||||
|         real_id = compat_urllib_parse.unquote(base64.b64decode(encoded_id.encode('ascii')).decode('utf-8')) | ||||
|         video_url = 'rtmpe://video.infoq.com/cfx/st/' + real_id | ||||
|         playpath = 'mp4:' + real_id | ||||
|  | ||||
|         # Extract title | ||||
|         video_title = self._search_regex(r'contentTitle = "(.*?)";', | ||||
|             webpage, 'title') | ||||
|  | ||||
|         # Extract description | ||||
|         video_description = self._html_search_regex(r'<meta name="description" content="(.*)"(?:\s*/)?>', | ||||
|             webpage, 'description', fatal=False) | ||||
|  | ||||
|         video_filename = video_url.split('/')[-1] | ||||
|         video_filename = playpath.split('/')[-1] | ||||
|         video_id, extension = video_filename.split('.') | ||||
|  | ||||
|         http_base = self._search_regex( | ||||
|             r'EXPRESSINSTALL_SWF\s*=\s*"(https?://[^/"]+/)', webpage, | ||||
|             'HTTP base URL') | ||||
|  | ||||
|         formats = [{ | ||||
|             'format_id': 'rtmp', | ||||
|             'url': video_url, | ||||
|             'ext': extension, | ||||
|             'play_path': playpath, | ||||
|         }, { | ||||
|             'format_id': 'http', | ||||
|             'url': http_base + real_id, | ||||
|         }] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'title': video_title, | ||||
|             'ext': extension,  # Extension is always(?) mp4, but seems to be flv | ||||
|             'description': video_description, | ||||
|             'formats': formats, | ||||
|         } | ||||
|   | ||||
| @@ -89,7 +89,7 @@ class InstagramUserIE(InfoExtractor): | ||||
|                     'uploader': user.get('full_name'), | ||||
|                     'uploader_id': user.get('username'), | ||||
|                     'like_count': like_count, | ||||
|                     'upload_timestamp': int_or_none(it.get('created_time')), | ||||
|                     'timestamp': int_or_none(it.get('created_time')), | ||||
|                 }) | ||||
|  | ||||
|             if not page['items']: | ||||
|   | ||||
| @@ -33,14 +33,14 @@ class IviIE(InfoExtractor): | ||||
|         }, | ||||
|         # Serial's serie | ||||
|         { | ||||
|             'url': 'http://www.ivi.ru/watch/dezhurnyi_angel/74791', | ||||
|             'md5': '3e6cc9a848c1d2ebcc6476444967baa9', | ||||
|             'url': 'http://www.ivi.ru/watch/dvoe_iz_lartsa/9549', | ||||
|             'md5': '221f56b35e3ed815fde2df71032f4b3e', | ||||
|             'info_dict': { | ||||
|                 'id': '74791', | ||||
|                 'id': '9549', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Дежурный ангел - 1 серия', | ||||
|                 'duration': 2490, | ||||
|                 'thumbnail': 'http://thumbs.ivi.ru/f7.vcp.digitalaccess.ru/contents/8/e/bc2f6c2b6e5d291152fdd32c059141.jpg', | ||||
|                 'title': 'Двое из ларца - Серия 1', | ||||
|                 'duration': 2655, | ||||
|                 'thumbnail': 'http://thumbs.ivi.ru/f15.vcp.digitalaccess.ru/contents/8/4/0068dc0677041f3336b7c2baad8fc0.jpg', | ||||
|             }, | ||||
|             'skip': 'Only works from Russia', | ||||
|          } | ||||
|   | ||||
| @@ -14,7 +14,7 @@ class JukeboxIE(InfoExtractor): | ||||
|     _VALID_URL = r'^http://www\.jukebox?\..+?\/.+[,](?P<video_id>[a-z0-9\-]+)\.html' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.jukebox.es/kosheen/videoclip,pride,r303r.html', | ||||
|         'md5': '5dc6477e74b1e37042ac5acedd8413e5', | ||||
|         'md5': '1574e9b4d6438446d5b7dbcdf2786276', | ||||
|         'info_dict': { | ||||
|             'id': 'r303r', | ||||
|             'ext': 'flv', | ||||
|   | ||||
| @@ -1,9 +1,12 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import json | ||||
| import os | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_str, | ||||
|     ExtractorError, | ||||
|     formatSeconds, | ||||
| ) | ||||
| @@ -24,34 +27,31 @@ class JustinTVIE(InfoExtractor): | ||||
|         /?(?:\#.*)?$ | ||||
|         """ | ||||
|     _JUSTIN_PAGE_LIMIT = 100 | ||||
|     IE_NAME = u'justin.tv' | ||||
|     IE_NAME = 'justin.tv' | ||||
|     IE_DESC = 'justin.tv and twitch.tv' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.twitch.tv/thegamedevhub/b/296128360', | ||||
|         u'file': u'296128360.flv', | ||||
|         u'md5': u'ecaa8a790c22a40770901460af191c9a', | ||||
|         u'info_dict': { | ||||
|             u"upload_date": u"20110927",  | ||||
|             u"uploader_id": 25114803,  | ||||
|             u"uploader": u"thegamedevhub",  | ||||
|             u"title": u"Beginner Series - Scripting With Python Pt.1" | ||||
|         'url': 'http://www.twitch.tv/thegamedevhub/b/296128360', | ||||
|         'md5': 'ecaa8a790c22a40770901460af191c9a', | ||||
|         'info_dict': { | ||||
|             'id': '296128360', | ||||
|             'ext': 'flv', | ||||
|             'upload_date': '20110927', | ||||
|             'uploader_id': 25114803, | ||||
|             'uploader': 'thegamedevhub', | ||||
|             'title': 'Beginner Series - Scripting With Python Pt.1' | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def report_download_page(self, channel, offset): | ||||
|         """Report attempt to download a single page of videos.""" | ||||
|         self.to_screen(u'%s: Downloading video information from %d to %d' % | ||||
|                 (channel, offset, offset + self._JUSTIN_PAGE_LIMIT)) | ||||
|  | ||||
|     # Return count of items, list of *valid* items | ||||
|     def _parse_page(self, url, video_id): | ||||
|         info_json = self._download_webpage(url, video_id, | ||||
|                                            u'Downloading video info JSON', | ||||
|                                            u'unable to download video info JSON') | ||||
|                                            'Downloading video info JSON', | ||||
|                                            'unable to download video info JSON') | ||||
|  | ||||
|         response = json.loads(info_json) | ||||
|         if type(response) != list: | ||||
|             error_text = response.get('error', 'unknown error') | ||||
|             raise ExtractorError(u'Justin.tv API: %s' % error_text) | ||||
|             raise ExtractorError('Justin.tv API: %s' % error_text) | ||||
|         info = [] | ||||
|         for clip in response: | ||||
|             video_url = clip['video_file_url'] | ||||
| @@ -62,7 +62,7 @@ class JustinTVIE(InfoExtractor): | ||||
|                 video_id = clip['id'] | ||||
|                 video_title = clip.get('title', video_id) | ||||
|                 info.append({ | ||||
|                     'id': video_id, | ||||
|                     'id': compat_str(video_id), | ||||
|                     'url': video_url, | ||||
|                     'title': video_title, | ||||
|                     'uploader': clip.get('channel_name', video_uploader_id), | ||||
| @@ -74,8 +74,6 @@ class JustinTVIE(InfoExtractor): | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         if mobj is None: | ||||
|             raise ExtractorError(u'invalid URL: %s' % url) | ||||
|  | ||||
|         api_base = 'http://api.justin.tv' | ||||
|         paged = False | ||||
| @@ -89,40 +87,41 @@ class JustinTVIE(InfoExtractor): | ||||
|             webpage = self._download_webpage(url, chapter_id) | ||||
|             m = re.search(r'PP\.archive_id = "([0-9]+)";', webpage) | ||||
|             if not m: | ||||
|                 raise ExtractorError(u'Cannot find archive of a chapter') | ||||
|                 raise ExtractorError('Cannot find archive of a chapter') | ||||
|             archive_id = m.group(1) | ||||
|  | ||||
|             api = api_base + '/broadcast/by_chapter/%s.xml' % chapter_id | ||||
|             doc = self._download_xml(api, chapter_id, | ||||
|                                              note=u'Downloading chapter information', | ||||
|                                              errnote=u'Chapter information download failed') | ||||
|             doc = self._download_xml( | ||||
|                 api, chapter_id, | ||||
|                 note='Downloading chapter information', | ||||
|                 errnote='Chapter information download failed') | ||||
|             for a in doc.findall('.//archive'): | ||||
|                 if archive_id == a.find('./id').text: | ||||
|                     break | ||||
|             else: | ||||
|                 raise ExtractorError(u'Could not find chapter in chapter information') | ||||
|                 raise ExtractorError('Could not find chapter in chapter information') | ||||
|  | ||||
|             video_url = a.find('./video_file_url').text | ||||
|             video_ext = video_url.rpartition('.')[2] or u'flv' | ||||
|             video_ext = video_url.rpartition('.')[2] or 'flv' | ||||
|  | ||||
|             chapter_api_url = u'https://api.twitch.tv/kraken/videos/c' + chapter_id | ||||
|             chapter_info_json = self._download_webpage(chapter_api_url, u'c' + chapter_id, | ||||
|                                    note='Downloading chapter metadata', | ||||
|                                    errnote='Download of chapter metadata failed') | ||||
|             chapter_info = json.loads(chapter_info_json) | ||||
|             chapter_api_url = 'https://api.twitch.tv/kraken/videos/c' + chapter_id | ||||
|             chapter_info = self._download_json( | ||||
|                 chapter_api_url, 'c' + chapter_id, | ||||
|                 note='Downloading chapter metadata', | ||||
|                 errnote='Download of chapter metadata failed') | ||||
|  | ||||
|             bracket_start = int(doc.find('.//bracket_start').text) | ||||
|             bracket_end = int(doc.find('.//bracket_end').text) | ||||
|  | ||||
|             # TODO determine start (and probably fix up file) | ||||
|             #  youtube-dl -v http://www.twitch.tv/firmbelief/c/1757457 | ||||
|             #video_url += u'?start=' + TODO:start_timestamp | ||||
|             #video_url += '?start=' + TODO:start_timestamp | ||||
|             # bracket_start is 13290, but we want 51670615 | ||||
|             self._downloader.report_warning(u'Chapter detected, but we can just download the whole file. ' | ||||
|                                             u'Chapter starts at %s and ends at %s' % (formatSeconds(bracket_start), formatSeconds(bracket_end))) | ||||
|             self._downloader.report_warning('Chapter detected, but we can just download the whole file. ' | ||||
|                                             'Chapter starts at %s and ends at %s' % (formatSeconds(bracket_start), formatSeconds(bracket_end))) | ||||
|  | ||||
|             info = { | ||||
|                 'id': u'c' + chapter_id, | ||||
|                 'id': 'c' + chapter_id, | ||||
|                 'url': video_url, | ||||
|                 'ext': video_ext, | ||||
|                 'title': chapter_info['title'], | ||||
| @@ -131,14 +130,12 @@ class JustinTVIE(InfoExtractor): | ||||
|                 'uploader': chapter_info['channel']['display_name'], | ||||
|                 'uploader_id': chapter_info['channel']['name'], | ||||
|             } | ||||
|             return [info] | ||||
|             return info | ||||
|         else: | ||||
|             video_id = mobj.group('videoid') | ||||
|             api = api_base + '/broadcast/by_archive/%s.json' % video_id | ||||
|  | ||||
|         self.report_extraction(video_id) | ||||
|  | ||||
|         info = [] | ||||
|         entries = [] | ||||
|         offset = 0 | ||||
|         limit = self._JUSTIN_PAGE_LIMIT | ||||
|         while True: | ||||
| @@ -146,8 +143,12 @@ class JustinTVIE(InfoExtractor): | ||||
|                 self.report_download_page(video_id, offset) | ||||
|             page_url = api + ('?offset=%d&limit=%d' % (offset, limit)) | ||||
|             page_count, page_info = self._parse_page(page_url, video_id) | ||||
|             info.extend(page_info) | ||||
|             entries.extend(page_info) | ||||
|             if not paged or page_count != limit: | ||||
|                 break | ||||
|             offset += limit | ||||
|         return info | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|             'id': video_id, | ||||
|             'entries': entries, | ||||
|         } | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import os | ||||
| import re | ||||
|  | ||||
| @@ -11,22 +13,22 @@ from ..aes import ( | ||||
|     aes_decrypt_text | ||||
| ) | ||||
|  | ||||
|  | ||||
| class KeezMoviesIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>keezmovies\.com/video/.+?(?P<videoid>[0-9]+))(?:[/?&]|$)' | ||||
|     _VALID_URL = r'^https?://(?:www\.)?keezmovies\.com/video/.+?(?P<videoid>[0-9]+)(?:[/?&]|$)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.keezmovies.com/video/petite-asian-lady-mai-playing-in-bathtub-1214711', | ||||
|         u'file': u'1214711.mp4', | ||||
|         u'md5': u'6e297b7e789329923fcf83abb67c9289', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Petite Asian Lady Mai Playing In Bathtub", | ||||
|             u"age_limit": 18, | ||||
|         'url': 'http://www.keezmovies.com/video/petite-asian-lady-mai-playing-in-bathtub-1214711', | ||||
|         'file': '1214711.mp4', | ||||
|         'md5': '6e297b7e789329923fcf83abb67c9289', | ||||
|         'info_dict': { | ||||
|             'title': 'Petite Asian Lady Mai Playing In Bathtub', | ||||
|             'age_limit': 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('videoid') | ||||
|         url = 'http://www.' + mobj.group('url') | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
| @@ -38,10 +40,10 @@ class KeezMoviesIE(InfoExtractor): | ||||
|             embedded_url = mobj.group(1) | ||||
|             return self.url_result(embedded_url) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<h1 [^>]*>([^<]+)', webpage, u'title') | ||||
|         video_url = compat_urllib_parse.unquote(self._html_search_regex(r'video_url=(.+?)&', webpage, u'video_url')) | ||||
|         if webpage.find('encrypted=true')!=-1: | ||||
|             password = self._html_search_regex(r'video_title=(.+?)&', webpage, u'password') | ||||
|         video_title = self._html_search_regex(r'<h1 [^>]*>([^<]+)', webpage, 'title') | ||||
|         video_url = compat_urllib_parse.unquote(self._html_search_regex(r'video_url=(.+?)&', webpage, 'video_url')) | ||||
|         if 'encrypted=true' in webpage: | ||||
|             password = self._html_search_regex(r'video_title=(.+?)&', webpage, 'password') | ||||
|             video_url = aes_decrypt_text(video_url, password, 32).decode('utf-8') | ||||
|         path = compat_urllib_parse_urlparse(video_url).path | ||||
|         extension = os.path.splitext(path)[1][1:] | ||||
|   | ||||
| @@ -1,37 +1,39 @@ | ||||
| # encoding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class KickStarterIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://www\.kickstarter\.com/projects/(?P<id>\d*)/.*' | ||||
|     _VALID_URL = r'https?://www\.kickstarter\.com/projects/(?P<id>[^/]*)/.*' | ||||
|     _TEST = { | ||||
|         u"url": u"https://www.kickstarter.com/projects/1404461844/intersection-the-story-of-josh-grant?ref=home_location", | ||||
|         u"file": u"1404461844.mp4", | ||||
|         u"md5": u"c81addca81327ffa66c642b5d8b08cab", | ||||
|         u"info_dict": { | ||||
|             u"title": u"Intersection: The Story of Josh Grant by Kyle Cowling", | ||||
|         'url': 'https://www.kickstarter.com/projects/1404461844/intersection-the-story-of-josh-grant?ref=home_location', | ||||
|         'md5': 'c81addca81327ffa66c642b5d8b08cab', | ||||
|         'info_dict': { | ||||
|             'id': '1404461844', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Intersection: The Story of Josh Grant by Kyle Cowling', | ||||
|             'description': 'A unique motocross documentary that examines the ' | ||||
|                 'life and mind of one of sports most elite athletes: Josh Grant.', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         m = re.match(self._VALID_URL, url) | ||||
|         video_id = m.group('id') | ||||
|         webpage_src = self._download_webpage(url, video_id) | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         video_url = self._search_regex(r'data-video="(.*?)">', | ||||
|             webpage_src, u'video URL') | ||||
|         if 'mp4' in video_url: | ||||
|             ext = 'mp4' | ||||
|         else: | ||||
|             ext = 'flv' | ||||
|         video_title = self._html_search_regex(r"<title>(.*?)</title>", | ||||
|             webpage_src, u'title').rpartition(u'\u2014 Kickstarter')[0].strip() | ||||
|         video_url = self._search_regex(r'data-video-url="(.*?)"', | ||||
|             webpage, 'video URL') | ||||
|         video_title = self._html_search_regex(r'<title>(.*?)</title>', | ||||
|             webpage, 'title').rpartition('— Kickstarter')[0].strip() | ||||
|  | ||||
|         results = [{ | ||||
|                     'id': video_id, | ||||
|                     'url': video_url, | ||||
|                     'title': video_title, | ||||
|                     'ext': ext, | ||||
|                     }] | ||||
|         return results | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'title': video_title, | ||||
|             'description': self._og_search_description(webpage), | ||||
|             'thumbnail': self._og_search_thumbnail(webpage), | ||||
|         } | ||||
|   | ||||
							
								
								
									
										35
									
								
								youtube_dl/extractor/ku6.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								youtube_dl/extractor/ku6.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class Ku6IE(InfoExtractor): | ||||
|     _VALID_URL = r'http://v\.ku6\.com/show/(?P<id>[a-zA-Z0-9\-\_]+)(?:\.)*html' | ||||
|     _TEST = { | ||||
|         'url': 'http://v.ku6.com/show/JG-8yS14xzBr4bCn1pu0xw...html', | ||||
|         'md5': '01203549b9efbb45f4b87d55bdea1ed1', | ||||
|         'info_dict': { | ||||
|             'id': 'JG-8yS14xzBr4bCn1pu0xw', | ||||
|             'ext': 'f4v', | ||||
|             'title': 'techniques test', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         title = self._search_regex(r'<h1 title=.*>(.*?)</h1>', webpage, 'title') | ||||
|         dataUrl = 'http://v.ku6.com/fetchVideo4Player/%s.html' % video_id | ||||
|         jsonData = self._download_json(dataUrl, video_id) | ||||
|         downloadUrl = jsonData['data']['f'] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'url': downloadUrl | ||||
|         } | ||||
|  | ||||
| @@ -24,7 +24,7 @@ class LifeNewsIE(InfoExtractor): | ||||
|             'ext': 'mp4', | ||||
|             'title': 'МВД разыскивает мужчин, оставивших в IKEA сумку с автоматом', | ||||
|             'description': 'Камеры наблюдения гипермаркета зафиксировали троих мужчин, спрятавших оружейный арсенал в камере хранения.', | ||||
|             'thumbnail': 'http://lifenews.ru/static/posts/2014/1/126342/.video.jpg', | ||||
|             'thumbnail': 're:http://.*\.jpg', | ||||
|             'upload_date': '20140130', | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import json | ||||
|  | ||||
| @@ -6,31 +8,35 @@ from ..utils import ( | ||||
|     compat_urllib_parse_urlparse, | ||||
|     compat_urlparse, | ||||
|     xpath_with_ns, | ||||
|     compat_str, | ||||
|     orderedSet, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class LivestreamIE(InfoExtractor): | ||||
|     IE_NAME = u'livestream' | ||||
|     IE_NAME = 'livestream' | ||||
|     _VALID_URL = r'http://new\.livestream\.com/.*?/(?P<event_name>.*?)(/videos/(?P<id>\d+))?/?$' | ||||
|     _TEST = { | ||||
|         u'url': u'http://new.livestream.com/CoheedandCambria/WebsterHall/videos/4719370', | ||||
|         u'file': u'4719370.mp4', | ||||
|         u'md5': u'0d2186e3187d185a04b3cdd02b828836', | ||||
|         u'info_dict': { | ||||
|             u'title': u'Live from Webster Hall NYC', | ||||
|             u'upload_date': u'20121012', | ||||
|         'url': 'http://new.livestream.com/CoheedandCambria/WebsterHall/videos/4719370', | ||||
|         'md5': '53274c76ba7754fb0e8d072716f2292b', | ||||
|         'info_dict': { | ||||
|             'id': '4719370', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Live from Webster Hall NYC', | ||||
|             'upload_date': '20121012', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _extract_video_info(self, video_data): | ||||
|         video_url = video_data.get('progressive_url_hd') or video_data.get('progressive_url') | ||||
|         return {'id': video_data['id'], | ||||
|                 'url': video_url, | ||||
|                 'ext': 'mp4', | ||||
|                 'title': video_data['caption'], | ||||
|                 'thumbnail': video_data['thumbnail_url'], | ||||
|                 'upload_date': video_data['updated_at'].replace('-','')[:8], | ||||
|                 } | ||||
|         return { | ||||
|             'id': compat_str(video_data['id']), | ||||
|             'url': video_url, | ||||
|             'ext': 'mp4', | ||||
|             'title': video_data['caption'], | ||||
|             'thumbnail': video_data['thumbnail_url'], | ||||
|             'upload_date': video_data['updated_at'].replace('-', '')[:8], | ||||
|         } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
| @@ -40,43 +46,43 @@ class LivestreamIE(InfoExtractor): | ||||
|  | ||||
|         if video_id is None: | ||||
|             # This is an event page: | ||||
|             config_json = self._search_regex(r'window.config = ({.*?});', | ||||
|                 webpage, u'window config') | ||||
|             config_json = self._search_regex( | ||||
|                 r'window.config = ({.*?});', webpage, 'window config') | ||||
|             info = json.loads(config_json)['event'] | ||||
|             videos = [self._extract_video_info(video_data['data']) | ||||
|                 for video_data in info['feed']['data'] if video_data['type'] == u'video'] | ||||
|                 for video_data in info['feed']['data'] if video_data['type'] == 'video'] | ||||
|             return self.playlist_result(videos, info['id'], info['full_name']) | ||||
|         else: | ||||
|             og_video = self._og_search_video_url(webpage, name=u'player url') | ||||
|             og_video = self._og_search_video_url(webpage, 'player url') | ||||
|             query_str = compat_urllib_parse_urlparse(og_video).query | ||||
|             query = compat_urlparse.parse_qs(query_str) | ||||
|             api_url = query['play_url'][0].replace('.smil', '') | ||||
|             info = json.loads(self._download_webpage(api_url, video_id, | ||||
|                                                      u'Downloading video info')) | ||||
|             info = json.loads(self._download_webpage( | ||||
|                 api_url, video_id, 'Downloading video info')) | ||||
|             return self._extract_video_info(info) | ||||
|  | ||||
|  | ||||
| # The original version of Livestream uses a different system | ||||
| class LivestreamOriginalIE(InfoExtractor): | ||||
|     IE_NAME = u'livestream:original' | ||||
|     _VALID_URL = r'https?://www\.livestream\.com/(?P<user>[^/]+)/video\?.*?clipId=(?P<id>.*?)(&|$)' | ||||
|     IE_NAME = 'livestream:original' | ||||
|     _VALID_URL = r'''(?x)https?://www\.livestream\.com/ | ||||
|         (?P<user>[^/]+)/(?P<type>video|folder) | ||||
|         (?:\?.*?Id=|/)(?P<id>.*?)(&|$) | ||||
|         ''' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.livestream.com/dealbook/video?clipId=pla_8aa4a3f1-ba15-46a4-893b-902210e138fb', | ||||
|         u'info_dict': { | ||||
|             u'id': u'pla_8aa4a3f1-ba15-46a4-893b-902210e138fb', | ||||
|             u'ext': u'flv', | ||||
|             u'title': u'Spark 1 (BitCoin) with Cameron Winklevoss & Tyler Winklevoss of Winklevoss Capital', | ||||
|         'url': 'http://www.livestream.com/dealbook/video?clipId=pla_8aa4a3f1-ba15-46a4-893b-902210e138fb', | ||||
|         'info_dict': { | ||||
|             'id': 'pla_8aa4a3f1-ba15-46a4-893b-902210e138fb', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Spark 1 (BitCoin) with Cameron Winklevoss & Tyler Winklevoss of Winklevoss Capital', | ||||
|         }, | ||||
|         u'params': { | ||||
|         'params': { | ||||
|             # rtmp | ||||
|             u'skip_download': True, | ||||
|             'skip_download': True, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         user = mobj.group('user') | ||||
|     def _extract_video(self, user, video_id): | ||||
|         api_url = 'http://x{0}x.api.channel.livestream.com/2.0/clipdetails?extendedInfo=true&id={1}'.format(user, video_id) | ||||
|  | ||||
|         info = self._download_xml(api_url, video_id) | ||||
| @@ -84,7 +90,7 @@ class LivestreamOriginalIE(InfoExtractor): | ||||
|         ns = {'media': 'http://search.yahoo.com/mrss'} | ||||
|         thumbnail_url = item.find(xpath_with_ns('media:thumbnail', ns)).attrib['url'] | ||||
|         # Remove the extension and number from the path (like 1.jpg) | ||||
|         path = self._search_regex(r'(user-files/.+)_.*?\.jpg$', thumbnail_url, u'path') | ||||
|         path = self._search_regex(r'(user-files/.+)_.*?\.jpg$', thumbnail_url, 'path') | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
| @@ -94,3 +100,44 @@ class LivestreamOriginalIE(InfoExtractor): | ||||
|             'ext': 'flv', | ||||
|             'thumbnail': thumbnail_url, | ||||
|         } | ||||
|  | ||||
|     def _extract_folder(self, url, folder_id): | ||||
|         webpage = self._download_webpage(url, folder_id) | ||||
|         urls = orderedSet(re.findall(r'<a href="(https?://livestre\.am/.*?)"', webpage)) | ||||
|  | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|             'id': folder_id, | ||||
|             'entries': [{ | ||||
|                 '_type': 'url', | ||||
|                 'url': video_url, | ||||
|             } for video_url in urls], | ||||
|         } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         id = mobj.group('id') | ||||
|         user = mobj.group('user') | ||||
|         url_type = mobj.group('type') | ||||
|         if url_type == 'folder': | ||||
|             return self._extract_folder(url, id) | ||||
|         else: | ||||
|             return self._extract_video(user, id) | ||||
|  | ||||
|  | ||||
| # The server doesn't support HEAD request, the generic extractor can't detect | ||||
| # the redirection | ||||
| class LivestreamShortenerIE(InfoExtractor): | ||||
|     IE_NAME = 'livestream:shortener' | ||||
|     IE_DESC = False  # Do not list | ||||
|     _VALID_URL = r'https?://livestre\.am/(?P<id>.+)' | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         id = mobj.group('id') | ||||
|         webpage = self._download_webpage(url, id) | ||||
|  | ||||
|         return { | ||||
|             '_type': 'url', | ||||
|             'url': self._og_search_url(webpage), | ||||
|         } | ||||
|   | ||||
| @@ -2,7 +2,6 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import datetime | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
| @@ -10,28 +9,48 @@ from .common import InfoExtractor | ||||
| class MailRuIE(InfoExtractor): | ||||
|     IE_NAME = 'mailru' | ||||
|     IE_DESC = 'Видео@Mail.Ru' | ||||
|     _VALID_URL = r'http://(?:www\.)?my\.mail\.ru/video/.*#video=/?(?P<id>[^/]+/[^/]+/[^/]+/\d+)' | ||||
|     _VALID_URL = r'http://(?:www\.)?my\.mail\.ru/(?:video/.*#video=/?(?P<idv1>(?:[^/]+/){3}\d+)|(?:(?P<idv2prefix>(?:[^/]+/){2})video/(?P<idv2suffix>[^/]+/\d+))\.html)' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://my.mail.ru/video/top#video=/mail/sonypicturesrus/75/76', | ||||
|         'md5': 'dea205f03120046894db4ebb6159879a', | ||||
|         'info_dict': { | ||||
|             'id': '46301138', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Новый Человек-Паук. Высокое напряжение. Восстание Электро', | ||||
|             'upload_date': '20140224', | ||||
|             'uploader': 'sonypicturesrus', | ||||
|             'uploader_id': 'sonypicturesrus@mail.ru', | ||||
|             'duration': 184, | ||||
|         } | ||||
|     } | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             'url': 'http://my.mail.ru/video/top#video=/mail/sonypicturesrus/75/76', | ||||
|             'md5': 'dea205f03120046894db4ebb6159879a', | ||||
|             'info_dict': { | ||||
|                 'id': '46301138', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Новый Человек-Паук. Высокое напряжение. Восстание Электро', | ||||
|                 'timestamp': 1393232740, | ||||
|                 'upload_date': '20140224', | ||||
|                 'uploader': 'sonypicturesrus', | ||||
|                 'uploader_id': 'sonypicturesrus@mail.ru', | ||||
|                 'duration': 184, | ||||
|             }, | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://my.mail.ru/corp/hitech/video/news_hi-tech_mail_ru/1263.html', | ||||
|             'md5': '00a91a58c3402204dcced523777b475f', | ||||
|             'info_dict': { | ||||
|                 'id': '46843144', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Samsung Galaxy S5 Hammer Smash Fail Battery Explosion', | ||||
|                 'timestamp': 1397217632, | ||||
|                 'upload_date': '20140411', | ||||
|                 'uploader': 'hitech', | ||||
|                 'uploader_id': 'hitech@corp.mail.ru', | ||||
|                 'duration': 245, | ||||
|             }, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         video_id = mobj.group('idv1') | ||||
|  | ||||
|         if not video_id: | ||||
|             video_id = mobj.group('idv2prefix') + mobj.group('idv2suffix') | ||||
|  | ||||
|         video_data = self._download_json( | ||||
|             'http://videoapi.my.mail.ru/videos/%s.json?new=1' % video_id, video_id, 'Downloading video JSON') | ||||
|             'http://api.video.mail.ru/videos/%s.json?new=1' % video_id, video_id, 'Downloading video JSON') | ||||
|  | ||||
|         author = video_data['author'] | ||||
|         uploader = author['name'] | ||||
| @@ -40,10 +59,11 @@ class MailRuIE(InfoExtractor): | ||||
|         movie = video_data['movie'] | ||||
|         content_id = str(movie['contentId']) | ||||
|         title = movie['title'] | ||||
|         if title.endswith('.mp4'): | ||||
|             title = title[:-4] | ||||
|         thumbnail = movie['poster'] | ||||
|         duration = movie['duration'] | ||||
|  | ||||
|         upload_date = datetime.datetime.fromtimestamp(video_data['timestamp']).strftime('%Y%m%d') | ||||
|         view_count = video_data['views_count'] | ||||
|  | ||||
|         formats = [ | ||||
| @@ -57,7 +77,7 @@ class MailRuIE(InfoExtractor): | ||||
|             'id': content_id, | ||||
|             'title': title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'upload_date': upload_date, | ||||
|             'timestamp': video_data['timestamp'], | ||||
|             'uploader': uploader, | ||||
|             'uploader_id': uploader_id, | ||||
|             'duration': duration, | ||||
|   | ||||
| @@ -1,15 +1,18 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class MDRIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?P<domain>(?:https?://)?(?:www\.)?mdr\.de)/mediathek/(?:.*)/(?P<type>video|audio)(?P<video_id>[^/_]+)_.*' | ||||
|     _VALID_URL = r'^(?P<domain>https?://(?:www\.)?mdr\.de)/(?:.*)/(?P<type>video|audio)(?P<video_id>[^/_]+)(?:_|\.html)' | ||||
|      | ||||
|     # No tests, MDR regularily deletes its videos | ||||
|     _TEST = { | ||||
|         'url': 'http://www.mdr.de/fakt/video189002.html', | ||||
|         'only_matching': True, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         m = re.match(self._VALID_URL, url) | ||||
| @@ -19,9 +22,9 @@ class MDRIE(InfoExtractor): | ||||
|         # determine title and media streams from webpage | ||||
|         html = self._download_webpage(url, video_id) | ||||
|  | ||||
|         title = self._html_search_regex(r'<h2>(.*?)</h2>', html, u'title') | ||||
|         title = self._html_search_regex(r'<h[12]>(.*?)</h[12]>', html, 'title') | ||||
|         xmlurl = self._search_regex( | ||||
|             r'(/mediathek/(?:.+)/(?:video|audio)[0-9]+-avCustom.xml)', html, u'XML URL') | ||||
|             r'dataURL:\'(/(?:.+)/(?:video|audio)[0-9]+-avCustom.xml)', html, 'XML URL') | ||||
|  | ||||
|         doc = self._download_xml(domain + xmlurl, video_id) | ||||
|         formats = [] | ||||
| @@ -41,7 +44,7 @@ class MDRIE(InfoExtractor): | ||||
|             if vbr_el is None: | ||||
|                 format.update({ | ||||
|                     'vcodec': 'none', | ||||
|                     'format_id': u'%s-%d' % (media_type, abr), | ||||
|                     'format_id': '%s-%d' % (media_type, abr), | ||||
|                 }) | ||||
|             else: | ||||
|                 vbr = int(vbr_el.text) // 1000 | ||||
| @@ -49,12 +52,9 @@ class MDRIE(InfoExtractor): | ||||
|                     'vbr': vbr, | ||||
|                     'width': int(a.find('frameWidth').text), | ||||
|                     'height': int(a.find('frameHeight').text), | ||||
|                     'format_id': u'%s-%d' % (media_type, vbr), | ||||
|                     'format_id': '%s-%d' % (media_type, vbr), | ||||
|                 }) | ||||
|             formats.append(format) | ||||
|         if not formats: | ||||
|             raise ExtractorError(u'Could not find any valid formats') | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|   | ||||
| @@ -13,8 +13,9 @@ class MetacriticIE(InfoExtractor): | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.metacritic.com/game/playstation-4/infamous-second-son/trailers/3698222', | ||||
|         'file': '3698222.mp4', | ||||
|         'info_dict': { | ||||
|             'id': '3698222', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'inFamous: Second Son - inSide Sucker Punch: Smoke & Mirrors', | ||||
|             'description': 'Take a peak behind-the-scenes to see how Sucker Punch brings smoke into the universe of inFAMOUS Second Son on the PS4.', | ||||
|             'duration': 221, | ||||
|   | ||||
| @@ -4,9 +4,10 @@ import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     unified_strdate, | ||||
|     compat_urllib_parse, | ||||
|     ExtractorError, | ||||
|     int_or_none, | ||||
|     parse_iso8601, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -24,6 +25,10 @@ class MixcloudIE(InfoExtractor): | ||||
|             'uploader': 'Daniel Holbach', | ||||
|             'uploader_id': 'dholbach', | ||||
|             'upload_date': '20111115', | ||||
|             'timestamp': 1321359578, | ||||
|             'thumbnail': 're:https?://.*\.jpg', | ||||
|             'view_count': int, | ||||
|             'like_count': int, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
| @@ -51,10 +56,6 @@ class MixcloudIE(InfoExtractor): | ||||
|  | ||||
|         webpage = self._download_webpage(url, track_id) | ||||
|  | ||||
|         api_url = 'http://api.mixcloud.com/%s/%s/' % (uploader, cloudcast_name) | ||||
|         info = self._download_json( | ||||
|             api_url, track_id, 'Downloading cloudcast info') | ||||
|  | ||||
|         preview_url = self._search_regex( | ||||
|             r'\s(?:data-preview-url|m-preview)="(.+?)"', webpage, 'preview url') | ||||
|         song_url = preview_url.replace('/previews/', '/c/originals/') | ||||
| @@ -65,16 +66,41 @@ class MixcloudIE(InfoExtractor): | ||||
|             template_url = template_url.replace('.mp3', '.m4a').replace('originals/', 'm4a/64/') | ||||
|             final_song_url = self._get_url(template_url) | ||||
|         if final_song_url is None: | ||||
|             raise ExtractorError(u'Unable to extract track url') | ||||
|             raise ExtractorError('Unable to extract track url') | ||||
|  | ||||
|         PREFIX = ( | ||||
|             r'<div class="cloudcast-play-button-container"' | ||||
|             r'(?:\s+[a-zA-Z0-9-]+(?:="[^"]+")?)*?\s+') | ||||
|         title = self._html_search_regex( | ||||
|             PREFIX + r'm-title="([^"]+)"', webpage, 'title') | ||||
|         thumbnail = self._proto_relative_url(self._html_search_regex( | ||||
|             PREFIX + r'm-thumbnail-url="([^"]+)"', webpage, 'thumbnail', | ||||
|             fatal=False)) | ||||
|         uploader = self._html_search_regex( | ||||
|             PREFIX + r'm-owner-name="([^"]+)"', | ||||
|             webpage, 'uploader', fatal=False) | ||||
|         uploader_id = self._search_regex( | ||||
|             r'\s+"profile": "([^"]+)",', webpage, 'uploader id', fatal=False) | ||||
|         description = self._og_search_description(webpage) | ||||
|         like_count = int_or_none(self._search_regex( | ||||
|             r'<meta itemprop="interactionCount" content="UserLikes:([0-9]+)"', | ||||
|             webpage, 'like count', fatal=False)) | ||||
|         view_count = int_or_none(self._search_regex( | ||||
|             r'<meta itemprop="interactionCount" content="UserPlays:([0-9]+)"', | ||||
|             webpage, 'play count', fatal=False)) | ||||
|         timestamp = parse_iso8601(self._search_regex( | ||||
|             r'<time itemprop="dateCreated" datetime="([^"]+)">', | ||||
|             webpage, 'upload date')) | ||||
|  | ||||
|         return { | ||||
|             'id': track_id, | ||||
|             'title': info['name'], | ||||
|             'title': title, | ||||
|             'url': final_song_url, | ||||
|             'description': info.get('description'), | ||||
|             'thumbnail': info['pictures'].get('extra_large'), | ||||
|             'uploader': info['user']['name'], | ||||
|             'uploader_id': info['user']['username'], | ||||
|             'upload_date': unified_strdate(info['created_time']), | ||||
|             'view_count': info['play_count'], | ||||
|             'description': description, | ||||
|             'thumbnail': thumbnail, | ||||
|             'uploader': uploader, | ||||
|             'uploader_id': uploader_id, | ||||
|             'timestamp': timestamp, | ||||
|             'view_count': view_count, | ||||
|             'like_count': like_count, | ||||
|         } | ||||
|   | ||||
| @@ -14,7 +14,7 @@ from ..utils import ( | ||||
| class MooshareIE(InfoExtractor): | ||||
|     IE_NAME = 'mooshare' | ||||
|     IE_DESC = 'Mooshare.biz' | ||||
|     _VALID_URL = r'http://mooshare\.biz/(?P<id>[\da-z]{12})' | ||||
|     _VALID_URL = r'http://(?:www\.)?mooshare\.biz/(?P<id>[\da-z]{12})' | ||||
|  | ||||
|     _TESTS = [ | ||||
|         { | ||||
|   | ||||
							
								
								
									
										47
									
								
								youtube_dl/extractor/morningstar.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								youtube_dl/extractor/morningstar.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class MorningstarIE(InfoExtractor): | ||||
|     IE_DESC = 'morningstar.com' | ||||
|     _VALID_URL = r'https?://(?:www\.)?morningstar\.com/[cC]over/video[cC]enter\.aspx\?id=(?P<id>[0-9]+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.morningstar.com/cover/videocenter.aspx?id=615869', | ||||
|         'md5': '6c0acface7a787aadc8391e4bbf7b0f5', | ||||
|         'info_dict': { | ||||
|             'id': '615869', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Get Ahead of the Curve on 2013 Taxes', | ||||
|             'description': "Vanguard's Joel Dickson on managing higher tax rates for high-income earners and fund capital-gain distributions in 2013.", | ||||
|             'thumbnail': r're:^https?://.*m(?:orning)?star\.com/.+thumb\.jpg$' | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         title = self._html_search_regex( | ||||
|             r'<h1 id="titleLink">(.*?)</h1>', webpage, 'title') | ||||
|         video_url = self._html_search_regex( | ||||
|             r'<input type="hidden" id="hidVideoUrl" value="([^"]+)"', | ||||
|             webpage, 'video URL') | ||||
|         thumbnail = self._html_search_regex( | ||||
|             r'<input type="hidden" id="hidSnapshot" value="([^"]+)"', | ||||
|             webpage, 'thumbnail', fatal=False) | ||||
|         description = self._html_search_regex( | ||||
|             r'<div id="mstarDeck".*?>(.*?)</div>', | ||||
|             webpage, 'description', fatal=False) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'url': video_url, | ||||
|             'thumbnail': thumbnail, | ||||
|             'description': description, | ||||
|         } | ||||
							
								
								
									
										63
									
								
								youtube_dl/extractor/motorsport.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								youtube_dl/extractor/motorsport.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import hashlib | ||||
| import json | ||||
| import re | ||||
| import time | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_parse_qs, | ||||
|     compat_str, | ||||
|     int_or_none, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class MotorsportIE(InfoExtractor): | ||||
|     IE_DESC = 'motorsport.com' | ||||
|     _VALID_URL = r'http://www\.motorsport\.com/[^/?#]+/video/(?:[^/?#]+/)(?P<id>[^/]+)/(?:$|[?#])' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.motorsport.com/f1/video/main-gallery/red-bull-racing-2014-rules-explained/', | ||||
|         'md5': '5592cb7c5005d9b2c163df5ac3dc04e4', | ||||
|         'info_dict': { | ||||
|             'id': '7063', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Red Bull Racing: 2014 Rules Explained', | ||||
|             'duration': 207, | ||||
|             'description': 'A new clip from Red Bull sees Daniel Ricciardo and Sebastian Vettel explain the 2014 Formula One regulations – which are arguably the most complex the sport has ever seen.', | ||||
|             'uploader': 'rainiere', | ||||
|             'thumbnail': r're:^http://.*motorsport\.com/.+\.jpg$' | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         display_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|         flashvars_code = self._html_search_regex( | ||||
|             r'<embed id="player".*?flashvars="([^"]+)"', webpage, 'flashvars') | ||||
|         flashvars = compat_parse_qs(flashvars_code) | ||||
|         params = json.loads(flashvars['parameters'][0]) | ||||
|  | ||||
|         e = compat_str(int(time.time()) + 24 * 60 * 60) | ||||
|         base_video_url = params['location'] + '?e=' + e | ||||
|         s = 'h3hg713fh32' | ||||
|         h = hashlib.md5((s + base_video_url).encode('utf-8')).hexdigest() | ||||
|         video_url = base_video_url + '&h=' + h | ||||
|  | ||||
|         uploader = self._html_search_regex( | ||||
|             r'(?s)<span class="label">Video by: </span>(.*?)</a>', webpage, | ||||
|             'uploader', fatal=False) | ||||
|  | ||||
|         return { | ||||
|             'id': params['video_id'], | ||||
|             'display_id': display_id, | ||||
|             'title': params['title'], | ||||
|             'url': video_url, | ||||
|             'description': params.get('description'), | ||||
|             'thumbnail': params.get('main_thumb'), | ||||
|             'duration': int_or_none(params.get('duration')), | ||||
|             'uploader': uploader, | ||||
|         } | ||||
							
								
								
									
										45
									
								
								youtube_dl/extractor/moviezine.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								youtube_dl/extractor/moviezine.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class MoviezineIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://www\.moviezine\.se/video/(?P<id>[^?#]+)' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.moviezine.se/video/205866', | ||||
|         'info_dict': { | ||||
|             'id': '205866', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Oculus - Trailer 1', | ||||
|             'description': 'md5:40cc6790fc81d931850ca9249b40e8a4', | ||||
|             'thumbnail': 're:http://.*\.jpg', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         jsplayer = self._download_webpage('http://www.moviezine.se/api/player.js?video=%s' % video_id, video_id, 'Downloading js api player') | ||||
|  | ||||
|         formats =[{ | ||||
|             'format_id': 'sd', | ||||
|             'url': self._html_search_regex(r'file: "(.+?)",', jsplayer, 'file'), | ||||
|             'quality': 0, | ||||
|             'ext': 'mp4', | ||||
|         }] | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': self._search_regex(r'title: "(.+?)",', jsplayer, 'title'), | ||||
|             'thumbnail': self._search_regex(r'image: "(.+?)",', jsplayer, 'image'), | ||||
|             'formats': formats, | ||||
|             'description': self._og_search_description(webpage), | ||||
|         } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user