Compare commits
	
		
			693 Commits
		
	
	
		
			v0.9.3
			...
			064707ac23
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 064707ac23 | ||
|  | b9b4d57b25 | ||
|  | b4618739f3 | ||
|  | 22c6c82c0e | ||
|  | 19c875c18a | ||
|  | 12bbc0058c | ||
|  | 9c81806b95 | ||
|  | 45303005b8 | ||
|  | 881aa9adce | ||
|  | 4714be5697 | ||
|  | cb7f4a72d7 | ||
|  | fb63ff0db8 | ||
|  | c5f447e20b | ||
|  | b4d3769fd5 | ||
|  | b199d597b2 | ||
|  | debfb37a47 | ||
|  | 67fd6ffdf6 | ||
|  | e57265016e | ||
|  | cf4c22898c | ||
|  | 3bb99541e7 | ||
|  | 8c367af0ff | ||
|  | bc1e3edf17 | ||
|  | e488f4a7da | ||
|  | afad38d8e1 | ||
|  | e9804d4184 | ||
|  | a1b80a7abb | ||
|  | 803bfa7084 | ||
|  | d1cb866b44 | ||
|  | a298e0cf16 | ||
|  | 766b0125fb | ||
|  | 998fa43fb2 | ||
|  | ecc6edac5a | ||
|  | ea518ba4c9 | ||
|  | ffdf4222bf | ||
|  | a97ef67411 | ||
|  | 813219cd9c | ||
|  | bb1f7d9294 | ||
|  | 3d28c958d3 | ||
|  | 6b68916d74 | ||
|  | 12e752e681 | ||
|  | 1f342d0c71 | ||
|  | 5e86d4a48a | ||
|  | 0838f84859 | ||
|  | abc938eacd | ||
|  | 4d13cd2c0b | ||
|  | 8f8971c706 | ||
|  | 2703d9513a | ||
|  | 3dce83de93 | ||
|  | ef8e7d4251 | ||
|  | a131e1ae73 | ||
|  | 84a86bd7bd | ||
|  | adfb5886c9 | ||
|  | 8d237ea4ef | ||
|  | 513bc6eadf | ||
|  | 856962af63 | ||
|  | 7c68a29181 | ||
|  | 2f4e3f2bb1 | ||
|  | 0389b838bc | ||
|  | 441f53e382 | ||
|  | 83c45dcf40 | ||
|  | cc9d81a39e | ||
|  | edf14cfd84 | ||
|  | ee79969eda | ||
|  | dbb20b1fdc | ||
|  | beee209249 | ||
|  | d6876ce13b | ||
|  | ed05d16a31 | ||
|  | 3806f01d2f | ||
|  | 5b69ced1e8 | ||
|  | 6b07f1d8b9 | ||
|  | 700cf14a50 | ||
|  | 1b08243cd2 | ||
|  | a0b978004c | ||
|  | efc8776e70 | ||
|  | 915f9a3782 | ||
|  | e136d77ade | ||
|  | 04aec15833 | ||
|  | dd5e1024db | ||
|  | 31d13f8fae | ||
|  | 19b4d929e2 | ||
|  | 27e5d1baae | ||
|  | 3a0b9867bc | ||
|  | a9c681818a | ||
|  | d279c96dd5 | ||
|  | d30589d1fa | ||
|  | 47c744e5e2 | ||
|  | 708869ea93 | ||
|  | 8b47bf3e5d | ||
|  | a2930b4386 | ||
|  | 2dc93ed18b | ||
|  | 2bd08c8254 | ||
|  | 81278ed553 | ||
|  | 589cec66e1 | ||
|  | 281a20f56a | ||
|  | ae8d205dbe | ||
|  | 1e6222f46a | ||
|  | 4f2a24848e | ||
|  | e670c80971 | ||
|  | ba7572eddd | ||
|  | a5c6fac976 | ||
|  | 1293814c3a | ||
|  | 1b2aeb01ce | ||
|  | cab8abd1a0 | ||
|  | edda2386fb | ||
|  | b0ad5f6097 | ||
|  | 6862bd7be3 | ||
|  | bc551a63c2 | ||
|  | c9f11b924d | ||
|  | 3236ea5b97 | ||
|  | 794696d327 | ||
|  | 7345de149a | ||
|  | 4fdf0bbc57 | ||
|  | d17f741f97 | ||
|  | 4a898b3ff5 | ||
|  | 7f84ca8d0c | ||
|  | c3a974a495 | ||
|  | 5b57d49a3e | ||
|  | 7af83c04c0 | ||
|  | b5ba338f86 | ||
|  | 50bfeb92b2 | ||
|  | 8d41ea5bfd | ||
|  | b10b14c8e9 | ||
|  | 144e81bd46 | ||
|  | 230c849b60 | ||
|  | 466f27a8c5 | ||
|  | dc12e01fc7 | ||
|  | d0e9a7f693 | ||
|  | 1ba21e03c6 | ||
|  | bcc8b44bb5 | ||
|  | b01b371c66 | ||
|  | 94a0f6b3df | ||
|  | 5df10ecc31 | ||
|  | 56786406ec | ||
|  | a4268f36cf | ||
|  | 8e7afa2edf | ||
|  | f07122d446 | ||
|  | 78c307780b | ||
|  | ad705d544a | ||
|  | 77f28315c9 | ||
|  | e0754031ad | ||
|  | f97d36b41f | ||
|  | bb2afe8e40 | ||
|  | faa0383af3 | ||
|  | e1e988272b | ||
|  | b159f04a6b | ||
|  | d91a7ea9e3 | ||
|  | 8056f3399e | ||
|  | fd9aa7ee90 | ||
|  | 53c19f473b | ||
|  | 78b5f05729 | ||
|  | f689376830 | ||
|  | d244856b41 | ||
|  | 3cd0f3a9a7 | ||
|  | f480d68b57 | ||
|  | db2bda1f9b | ||
|  | f834c01921 | ||
|  | f945fa80b3 | ||
|  | 70faa86e34 | ||
|  | 61502ed32a | ||
|  | bfca20bb12 | ||
|  | 0fd86d05a1 | ||
|  | c688d64062 | ||
|  | 2f973f129d | ||
|  | 9b81365b0a | ||
|  | a079797fca | ||
|  | 6ab298f6e8 | ||
|  | a159999879 | ||
|  | a71835a5b8 | ||
|  | 86a6e07804 | ||
|  | 73c6be1969 | ||
|  | 7db7868d2b | ||
|  | 18ec1f5680 | ||
|  | 8e65074b11 | ||
|  | d720438aef | ||
|  | ec0e3a91d1 | ||
|  | 48e7203ca6 | ||
|  | 4f76b79629 | ||
|  | 1eeae78a9f | ||
|  | bc27f756ed | ||
|  | 6302d5fb8b | ||
|  | 24e238c425 | ||
|  | 070f57fcc4 | ||
|  | a4ce45e9b0 | ||
|  | a3efa7702a | ||
|  | d7a5d00439 | ||
|  | 6636d49cc0 | ||
|  | 8e6ee4636e | ||
|  | 71f19dd3c7 | ||
|  | e166b472c5 | ||
|  | 28c867a115 | ||
|  | f20a04b2a0 | ||
|  | 1f961b2ca7 | ||
|  | e579e0c767 | ||
|  | 6693ec9c36 | ||
|  | 53856a3622 | ||
|  | 0b99238676 | ||
|  | cb2c68e25a | ||
|  | fd5553a9f5 | ||
|  | 60ebbd87d8 | ||
|  | 3a5185fcc8 | ||
|  | ce469d5e5a | ||
|  | 4f0f126e48 | ||
|  | 94c30a2440 | ||
|  | 1460b2f421 | ||
|  | 968223690e | ||
|  | 789d9d8ca1 | ||
|  | 2ce99a2c44 | ||
|  | ee207e994f | ||
|  | c374aca890 | ||
|  | c28ca58537 | ||
|  | 0578ea2c3c | ||
|  | e51ce99c1a | ||
|  | 3440039610 | ||
|  | 279f637c75 | ||
|  | d940b64517 | ||
|  | 403870e39e | ||
|  | 0383d613e6 | ||
|  | 40e9825ee0 | ||
|  | ab9ca94181 | ||
|  | 0f99a23af7 | ||
|  | bc5163adaf | ||
|  | 0561718917 | ||
|  | c1861627fb | ||
|  | e5eccab871 | ||
|  | 27f76ba659 | ||
|  | 589117b9e7 | ||
|  | 80300cd160 | ||
|  | 76171408cc | ||
|  | c1800a174f | ||
|  | 8ae8435940 | ||
|  | f916cb3b53 | ||
|  | 929c2137bf | ||
|  | 98056e91c5 | ||
|  | 944a7248c3 | ||
|  | caa2ecd0b7 | ||
|  | dfc2d0652f | ||
|  | 8d25540445 | ||
|  | 6ea174bfd4 | ||
|  | 56e43aec0e | ||
|  | 491d120c25 | ||
|  | 82d071d52c | ||
|  | 8190654a91 | ||
|  | 5e21702d16 | ||
|  | 3df4172237 | ||
|  | e0710a2ec1 | ||
|  | d20fc3b9ce | ||
|  | f25faec108 | ||
|  | 2750658c3c | ||
|  | e6bc5bbab3 | ||
|  | de5f3a9d9e | ||
|  | 7f0da012c2 | ||
|  | 76ecbf5eb0 | ||
|  | 06881a4c70 | ||
|  | c14fdd82db | ||
|  | b1a02ad930 | ||
|  | 2b580c60e9 | ||
|  | 27ffba3b14 | ||
|  | fb7bf437ba | ||
|  | d8baf0b9e7 | ||
|  | a6945fe880 | ||
|  | 6ff77dd8c7 | ||
|  | 1d925a608b | ||
|  | 646669ca75 | ||
|  | 0ec2baaa83 | ||
|  | 5abaaefd1c | ||
|  | 687afea0f2 | ||
|  | 7398d4fa2b | ||
|  | d73c8c3627 | ||
|  | f921b91c5b | ||
|  | 8ed3c1b159 | ||
|  | 4f947cdbb5 | ||
|  | ec6c29052a | ||
|  | 6b117502f3 | ||
|  | a367aa0b31 | ||
|  | 7f6843df55 | ||
|  | 4b485d54b6 | ||
|  | e80a040db4 | ||
|  | c357fd085b | ||
|  | d0c5f29b0a | ||
|  | 3e7b20c379 | ||
|  | f4a997c0ef | ||
|  | 102e74bb63 | ||
|  | 84fa15e44c | ||
|  | 7b8ecf8fe3 | ||
|  | 79ebf920ea | ||
|  | 0d05d42f70 | ||
|  | 95989b6da7 | ||
|  | 22e57f99a1 | ||
|  | b9d29c0417 | ||
|  | edc33db9e8 | ||
|  | 45d8b45d96 | ||
|  | b6a6d7dc68 | ||
|  | c57b84cd0b | ||
|  | 78e7841b5e | ||
|  | e41d981449 | ||
|  | 381227af66 | ||
|  | 2f8d0728ba | ||
|  | 13bfc5f2f9 | ||
|  | f8d3b571ba | ||
|  | 64b1e52d4c | ||
|  | b650f7ee9a | ||
|  | d4446280c7 | ||
|  | 3443a233f4 | ||
|  | 861f17bc4d | ||
|  | 41bbe18e3d | ||
|  | 5f9c357a15 | ||
|  | c089298f46 | ||
|  | be968e0caa | ||
|  | d32b7b612a | ||
|  | 160386be62 | ||
|  | 64bdde8f33 | ||
|  | 8739318101 | ||
|  | 1ac569badd | ||
|  | e38f891693 | ||
|  | 89a277c354 | ||
|  | 8238387c7d | ||
|  | 6c829581af | ||
|  | d180650c1b | ||
|  | 772bf5518f | ||
|  | 153dc0bdad | ||
|  | b7ea8e6001 | ||
|  | b0bf5ba8e0 | ||
|  | 8169a5f776 | ||
|  | b4b8914448 | ||
|  | 2ea2c89b4a | ||
|  | 479ca59a6a | ||
|  | 343f987a78 | ||
|  | 492465a525 | ||
|  | f185e44f93 | ||
|  | 5f2c318baf | ||
|  | 531a5b77d0 | ||
|  | f9245cdfed | ||
|  | bad9c7a4b9 | ||
|  | 576e0949e0 | ||
|  | d807648d2b | ||
|  | 47ea88e025 | ||
|  | 345a473ee0 | ||
|  | c6dc432d06 | ||
|  | af3bd55535 | ||
|  | 5fa1d86191 | ||
|  | d4859b675a | ||
|  | 9aa427031e | ||
|  | 9e8fe7bc1e | ||
|  | 90813c959d | ||
|  | 940a65954c | ||
|  | 9b4e753a79 | ||
|  | e0be9029e4 | ||
|  | 0ae213c240 | ||
|  | 08117e7a54 | ||
|  | 51c3226070 | ||
|  | 5396d19d7d | ||
|  | 11501e6899 | ||
|  | 4eb49b9119 | ||
|  | 4c2da22750 | ||
|  | 753b9cbae2 | ||
|  | 2c73cabe22 | ||
|  | d6ca091b7b | ||
|  | aa3faca246 | ||
|  | f0e849e9c0 | ||
|  | ddcbd6a790 | ||
|  | 28e3b6285e | ||
|  | 348db90f7b | ||
|  | 0d780b9b80 | ||
|  | 8ab718becd | ||
|  | 1943c357fa | ||
|  | 3be0d8389b | ||
|  | d7d1c83276 | ||
|  | 8591e2ffd5 | ||
|  | c2225bf2fd | ||
|  | 0617d7b49f | ||
|  | 42b288ee98 | ||
|  | ead7203e40 | ||
|  | bd2b947255 | ||
|  | f367bd2d0d | ||
|  | a8ce44b109 | ||
|  | 3b43d3f0bd | ||
|  | 06da486140 | ||
|  | a24a7d5636 | ||
|  | bc197fd665 | ||
|  | e35cc71cf4 | ||
|  | 7aa774b4ef | ||
|  | 9bb2de79fa | ||
|  | 21246144ab | ||
|  | 0e0845914b | ||
|  | 778e827277 | ||
|  | f36d4fa38d | ||
|  | 5b89c2d504 | ||
|  | 49b213bb2d | ||
|  | aed75c7d1b | ||
|  | ac51e4e4d5 | ||
|  | d8d84ae629 | ||
|  | 3f75f8ed31 | ||
|  | 8aef4dc2ec | ||
|  | b1e7ec706b | ||
|  | b5cd780360 | ||
|  | a8da94ee6d | ||
|  | f564c732d4 | ||
|  | 8beb1e5753 | ||
|  | d98d802a33 | ||
|  | d750f29fad | ||
|  | f425d32846 | ||
|  | 043d6b492d | ||
|  | 0bcccfa65e | ||
|  | 0716b1b8d8 | ||
|  | 47168e682d | ||
|  | 718d864dc8 | ||
|  | 22a691ec0f | ||
|  | dfcc826b7e | ||
|  | d1ee664ef5 | ||
|  | abcc6518bb | ||
|  | 2ef9ec3358 | ||
|  | f84cf3bf2d | ||
|  | bdcc2d2fa4 | ||
|  | 7e8e7f15a4 | ||
|  | 1ca3ad6237 | ||
|  | f3c878d949 | ||
|  | ee0c30ebb1 | ||
|  | c2f0c908d9 | ||
|  | 3edaaa0400 | ||
|  | 21a443baf2 | ||
|  | f6f47b5500 | ||
|  | 920c724656 | ||
|  | e50b814e07 | ||
|  | 2294082168 | ||
|  | 2661a28936 | ||
|  | 31a6834b1f | ||
|  | f66d98bcfe | ||
|  | ed7466621f | ||
|  | ead450aeb8 | ||
|  | d934cefa8b | ||
|  | 41807837b8 | ||
|  | 4419c816f5 | ||
|  | 4993da727a | ||
|  | 86a163e337 | ||
|  | c2fb602bee | ||
|  | f565d6f31a | ||
|  | 5af01bb8ff | ||
|  | 714e783e0d | ||
|  | fb1b0afddb | ||
|  | e6fdc56d25 | ||
|  | 5b965e63f8 | ||
|  | af86550e71 | ||
|  | e57ae069a7 | ||
|  | 39adc646e6 | ||
|  | 0947e77082 | ||
|  | 637b0ded09 | ||
|  | 9b7a84ea45 | ||
|  | ead696cbad | ||
|  | da23ad5eb5 | ||
|  | b63a0dfa01 | ||
|  | 6c00724a84 | ||
|  | 7619224809 | ||
|  | e0d3dd9050 | ||
|  | 71bf5e0e4f | ||
|  | 540e530420 | ||
|  | 070a8cad15 | ||
|  | 5d094b38b0 | ||
|  | af3d385ff5 | ||
|  | c352a0d698 | ||
|  | 060f64b4d2 | ||
|  | 4f032cd946 | ||
|  | cee6039ec3 | ||
|  | c8f8b818e0 | ||
|  | 08922ae284 | ||
|  | 51d606a54e | ||
|  | 2b76d71c67 | ||
|  | 67edd19eb8 | ||
|  | eaaa526cfc | ||
|  | 843c0f6c37 | ||
|  | 44ebf38e47 | ||
|  | d640e7d2ea | ||
|  | 66736519ed | ||
|  | 73f4c98be9 | ||
|  | b2ff7fefaa | ||
|  | 2edb95dfdd | ||
|  | e0bb9960fb | ||
|  | 71608845c0 | ||
|  | 0048e82151 | ||
|  | 6116bc9ca4 | ||
|  | c7cbbdd1c8 | ||
|  | b599033c54 | ||
|  | 7bf6a9fadc | ||
|  | 4490360e11 | ||
|  | a4dfe0d279 | ||
|  | 47679d1d3b | ||
|  | 62e17daf78 | ||
|  | 1f359f2a72 | ||
|  | cebe7a28c0 | ||
|  | 91778f43b7 | ||
|  | e3602e83ce | ||
|  | 36742bf30b | ||
|  | e614800d5f | ||
|  | 151a114235 | ||
|  | c842be3a52 | ||
|  | a264fac2b4 | ||
|  | 0767ef4902 | ||
|  | abe3357e67 | ||
|  | 19457efe9b | ||
|  | 487a2eb3e3 | ||
|  | 38f66147cb | ||
|  | ffa26c20b5 | ||
|  | 430ada7f84 | ||
|  | 988e37eb42 | ||
|  | 1938b90bce | ||
|  | f61d1403f3 | ||
|  | d228f34f64 | ||
|  | 97049556ed | ||
|  | b64c6a94cc | ||
|  | edc655bae7 | ||
|  | 884af48270 | ||
|  | 95f018fad3 | ||
|  | b44758a195 | ||
|  | f1c20d490e | ||
|  | 04372d498e | ||
|  | 63ea899605 | ||
|  | 4fdd145d1e | ||
|  | 57ee68b0e0 | ||
|  | 99c6884681 | ||
|  | 1c1438e9bc | ||
|  | 22f1b3e489 | ||
|  | fb1ad5800c | ||
|  | 4dd15b05ef | ||
|  | d7cdb644c4 | ||
|  | bfcf4950b3 | ||
|  | 6612c97f05 | ||
|  | b92cf62726 | ||
|  | a53ba33a81 | ||
|  | c04d38cf63 | ||
|  | a051adcbc0 | ||
|  | 900a9cdf72 | ||
|  | 611b329934 | ||
|  | 2642788bc1 | ||
|  | 8268445f0b | ||
|  | c12dcd9263 | ||
|  | 3142524809 | ||
|  | 4c9d3bd9d7 | ||
|  | ba103066b8 | ||
|  | 0b0d6179a2 | ||
|  | e8806d4ef8 | ||
|  | c96e5f174c | ||
|  | 315242e069 | ||
|  | a94fa5fbe3 | ||
|  | 90203afdd0 | ||
|  | 2c0d098852 | ||
|  | e4290cd465 | ||
|  | 46b85dec5c | ||
|  | bbc34bd009 | ||
|  | c495317e65 | ||
|  | a946050228 | ||
|  | 83789dcefa | ||
|  | 4f1f9bf1ce | ||
|  | 32c72c2f35 | ||
|  | 42ae0035af | ||
|  | 96e28fdbe6 | ||
|  | 0f889f50cf | ||
|  | 478eaebdec | ||
|  | 7ecf229db5 | ||
|  | dda75c6099 | ||
|  | 28d5ac9f90 | ||
|  | 52acfb4636 | ||
|  | 2a64bad385 | ||
|  | 1a73699f1a | ||
|  | 1b5a7a0063 | ||
|  | 4b3eb440cf | ||
|  | d1f457866b | ||
|  | 6f29aa82cb | ||
|  | b1a2ff7d84 | ||
|  | 883b16e251 | ||
|  | 116b39cf6a | ||
|  | eae1db9c7d | ||
|  | 730bab5d40 | ||
|  | d52dac233e | ||
|  | 1f37277a8d | ||
|  | 15014d7055 | ||
|  | 7a35ca05b1 | ||
|  | be6b6909d9 | ||
|  | 42c1d26b2e | ||
|  | d38f8ad2ec | ||
|  | 023fd58f05 | ||
|  | ad10a8f07f | ||
|  | 7d6cf039d4 | ||
|  | f0271e17b0 | ||
|  | 57954816b2 | ||
|  | 3e4e1f9bb9 | ||
|  | 7340918209 | ||
|  | 707df4f941 | ||
|  | 8eb6b83411 | ||
|  | e0aedd617b | ||
|  | ee81620c14 | ||
|  | 2d027af71a | ||
|  | 9d5f06b810 | ||
|  | b8fdcda2fb | ||
|  | 0dac7b7b81 | ||
|  | b750e753d6 | ||
|  | ee33e92bed | ||
|  | 7413a643f6 | ||
|  | 34452f9220 | ||
|  | 24831b2462 | ||
|  | cd4a18cb5a | ||
|  | c00b3df8b2 | ||
|  | 1beb821b2c | ||
|  | a58791048a | ||
|  | f0c6e8612f | ||
|  | 1cebbf92e6 | ||
|  | a64982583b | ||
|  | cb8b0915de | ||
|  | 1d2576b06d | ||
|  | ead9a3c0e9 | ||
|  | 59ba418faa | ||
|  | c51a332560 | ||
|  | a73d2feed6 | ||
|  | 6929193e9d | ||
|  | fea4ad9e89 | ||
|  | 68099049d4 | ||
|  | 44cf08bdfd | ||
|  | 9e32cf17a4 | ||
|  | 0661367ebb | ||
|  | 3c07e42ba2 | ||
|  | 2cd6376818 | ||
|  | 5e7f7750de | ||
|  | 2a223ec6db | ||
|  | a99108fff6 | ||
|  | 8de4698cc4 | ||
|  | 637319ec2c | ||
|  | f9398564cd | ||
|  | b57f423eb4 | ||
|  | 3093f1f2b6 | ||
|  | 961777e0c1 | ||
|  | d7139701f7 | ||
|  | c6bac17d48 | ||
|  | 3638fc5356 | ||
|  | aca9176f7f | ||
|  | 0d5e4f6d3f | ||
|  | 92a5ffdef8 | ||
|  | b3359fccdb | ||
|  | d8f7366d1f | ||
|  | ff94dc20af | ||
|  | a8df0a548f | ||
|  | 13d0dc4ba4 | ||
|  | 64125a1aca | ||
|  | 4feae03092 | ||
|  | 5f993c2bf8 | ||
|  | 35bbcbffba | ||
|  | 5faca54d67 | ||
|  | 82496b8e04 | ||
|  | 2d74ec7823 | ||
|  | 1d42c4d3a6 | ||
|  | 4a8ef00442 | ||
|  | add06ffa7a | ||
|  | fbb8d8e24a | ||
|  | cd0e001219 | ||
|  | bf53f4fc74 | ||
|  | 11e59e023c | ||
|  | c81d7d2bfb | ||
|  | 0885796fa8 | ||
|  | 1279481b62 | ||
|  | 5fb3412915 | ||
|  | c708a5ecf6 | ||
|  | e4f29e5f2b | ||
|  | 779fa409e5 | ||
|  | fc8c2dfa14 | ||
|  | 9f7d308961 | ||
|  | 8dacc37ba9 | ||
|  | 39eafa5a3e | ||
|  | d2741ca419 | ||
|  | a76ebbb22a | ||
|  | 83a45ebc03 | ||
|  | 76c2c65a7b | ||
|  | 99a7d0d534 | ||
|  | 8e68531ce4 | ||
|  | ed7b8488cb | ||
|  | 386cb4a6c1 | ||
|  | c95544dcb0 | ||
|  | 4083348c40 | ||
|  | b1cccf4173 | ||
|  | e1e1a0d611 | ||
|  | fb88f8d459 | ||
|  | 0fdab3968d | ||
|  | ac0e72d167 | ||
|  | e8fbaefa72 | ||
|  | de21eafe7b | ||
|  | 44b3b1330a | ||
|  | fd2e554b98 | ||
|  | d7acb9a40d | ||
|  | f63b9d7c4a | ||
|  | 7a0c64bf9e | ||
|  | ef352f097a | ||
|  | 357083efce | ||
|  | 0d75c09036 | ||
|  | 58c7e08d12 | ||
|  | b5443daeb1 | ||
|  | 5da3e5e4bf | ||
|  | f4dec2e48e | 
							
								
								
									
										7
									
								
								.bumpversion.cfg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | |||||||
|  | [bumpversion] | ||||||
|  | current_version = 1.9.6 | ||||||
|  | commit = True | ||||||
|  | tag = True | ||||||
|  |  | ||||||
|  | [bumpversion:file:fbchat/__init__.py] | ||||||
|  |  | ||||||
							
								
								
									
										34
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | |||||||
|  | --- | ||||||
|  | name: Bug report | ||||||
|  | about: Create a report if you're having trouble with `fbchat` | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | ## Description of the problem | ||||||
|  | Example: Logging in fails when the character `%` is in the password. A specific password that fails is `a_password_with_%` | ||||||
|  |  | ||||||
|  | ## Code to reproduce | ||||||
|  | ```py | ||||||
|  | # Example code | ||||||
|  | from fbchat import Client | ||||||
|  | client = Client("[REDACTED_USERNAME]", "a_password_with_%") | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Traceback | ||||||
|  | ``` | ||||||
|  | Traceback (most recent call last): | ||||||
|  |   File "<test.py>", line 1, in <module> | ||||||
|  |   File "[site-packages]/fbchat/client.py", line 78, in __init__ | ||||||
|  |     self.login(email, password, max_tries) | ||||||
|  |   File "[site-packages]/fbchat/client.py", line 407, in login | ||||||
|  |     raise FBchatUserError('Login failed. Check email/password. (Failed on URL: {})'.format(login_url)) | ||||||
|  | fbchat.models.FBchatUserError: Login failed. Check email/password. (Failed on URL: https://m.facebook.com/login.php?login_attempt=1) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Environment information | ||||||
|  | - Python version | ||||||
|  | - `fbchat` version | ||||||
|  | - If relevant, output from `$ python -m pip list` | ||||||
|  |  | ||||||
|  | If you have done any research, include that. | ||||||
|  | Make sure to redact all personal information. | ||||||
							
								
								
									
										19
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | |||||||
|  | --- | ||||||
|  | name: Feature request | ||||||
|  | about: Suggest a feature that you'd like to see implemented | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | ## Description | ||||||
|  | Example: There's no way to send messages to groups | ||||||
|  |  | ||||||
|  | ## Research (if applicable) | ||||||
|  | Example: I've found the URL `https://facebook.com/send_message.php`, to which you can send a POST requests with the following JSON: | ||||||
|  | ```json | ||||||
|  | { | ||||||
|  |    "text": message_content, | ||||||
|  |    "fbid": group_id, | ||||||
|  |    "some_variable": ? | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  | But I don't know how what `some_variable` does, and it doesn't work without it. I've found some examples of `some_variable` to be: `MTIzNDU2Nzg5MA`, `MTIzNDU2Nzg5MQ` and `MTIzNDU2Nzg5Mg` | ||||||
							
								
								
									
										16
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,14 +1,18 @@ | |||||||
| *py[co] | *py[co] | ||||||
|  |  | ||||||
|  | .idea/ | ||||||
|  |  | ||||||
| # Test scripts | # Test scripts | ||||||
| *.sh | *.sh | ||||||
|  |  | ||||||
| # Packages | # Packages | ||||||
| *.egg | *.egg | ||||||
| *.egg-info | *.egg-info | ||||||
|  | *.dist-info | ||||||
| dist | dist | ||||||
| build | build | ||||||
| eggs | eggs | ||||||
|  | .eggs | ||||||
| parts | parts | ||||||
| bin | bin | ||||||
| var | var | ||||||
| @@ -22,5 +26,13 @@ develop-eggs | |||||||
| # Sphinx documentation | # Sphinx documentation | ||||||
| docs/_build/ | docs/_build/ | ||||||
|  |  | ||||||
| # Data for tests | # Scripts and data for tests | ||||||
| tests.data | my_tests.py | ||||||
|  | my_test_data.json | ||||||
|  | my_data.json | ||||||
|  | tests.data | ||||||
|  | .pytest_cache | ||||||
|  |  | ||||||
|  | # Virtual environment | ||||||
|  | venv/ | ||||||
|  | .venv*/ | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								.readthedocs.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | |||||||
|  | # See https://docs.readthedocs.io/en/stable/config-file/v2.html for details | ||||||
|  | version: 2 | ||||||
|  |  | ||||||
|  | formats: | ||||||
|  |   - pdf | ||||||
|  |   - htmlzip | ||||||
|  |  | ||||||
|  | python: | ||||||
|  |   version: 3.6 | ||||||
|  |   install: | ||||||
|  |     - path: . | ||||||
|  |       extra_requirements: | ||||||
|  |         - docs | ||||||
|  |  | ||||||
|  | # Build documentation in the docs/ directory with Sphinx | ||||||
|  | sphinx: | ||||||
|  |   configuration: docs/conf.py | ||||||
|  |   fail_on_warning: true | ||||||
							
								
								
									
										61
									
								
								.travis.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,61 @@ | |||||||
|  | sudo: false | ||||||
|  | language: python | ||||||
|  | python: 3.6 | ||||||
|  |  | ||||||
|  | cache: pip | ||||||
|  |  | ||||||
|  | before_install: pip install flit | ||||||
|  | # Use `--deps production` so that we don't install unnecessary dependencies | ||||||
|  | install: flit install --deps production --extras test | ||||||
|  | script: pytest -m offline | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   include: | ||||||
|  |   - python: 2.7 | ||||||
|  |     before_install: | ||||||
|  |     - sudo apt-get -y install python3-pip python3-setuptools | ||||||
|  |     - sudo pip3 install flit | ||||||
|  |     install: flit install --python python --deps production --extras test | ||||||
|  |   - python: 3.4 | ||||||
|  |   - python: 3.5 | ||||||
|  |   - python: 3.6 | ||||||
|  |   - python: 3.7 | ||||||
|  |     dist: xenial | ||||||
|  |     sudo: required | ||||||
|  |   - python: pypy3.5 | ||||||
|  |  | ||||||
|  |   - name: Lint | ||||||
|  |     before_install: skip | ||||||
|  |     install: pip install black | ||||||
|  |     script: black --check --verbose . | ||||||
|  |  | ||||||
|  |   - stage: deploy | ||||||
|  |     name: GitHub Releases | ||||||
|  |     if: tag IS present | ||||||
|  |     install: skip | ||||||
|  |     script: flit build | ||||||
|  |     deploy: | ||||||
|  |       provider: releases | ||||||
|  |       api_key: $GITHUB_OAUTH_TOKEN | ||||||
|  |       file_glob: true | ||||||
|  |       file: dist/* | ||||||
|  |       skip_cleanup: true | ||||||
|  |       draft: false | ||||||
|  |       on: | ||||||
|  |         tags: true | ||||||
|  |  | ||||||
|  |   - stage: deploy | ||||||
|  |     name: PyPI | ||||||
|  |     if: tag IS present | ||||||
|  |     install: skip | ||||||
|  |     script: skip | ||||||
|  |     deploy: | ||||||
|  |       provider: script | ||||||
|  |       script: flit publish | ||||||
|  |       on: | ||||||
|  |         tags: true | ||||||
|  |  | ||||||
|  | notifications: | ||||||
|  |   email: | ||||||
|  |     on_success: never | ||||||
|  |     on_failure: change | ||||||
							
								
								
									
										75
									
								
								CODE_OF_CONDUCT
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,75 @@ | |||||||
|  | Contributor Covenant Code of Conduct | ||||||
|  |  | ||||||
|  | Our Pledge | ||||||
|  |  | ||||||
|  | In the interest of fostering an open and welcoming environment, we as | ||||||
|  | contributors and maintainers pledge to making participation in our project and | ||||||
|  | our community a harassment-free experience for everyone, regardless of age, body | ||||||
|  | size, disability, ethnicity, gender identity and expression, level of experience, | ||||||
|  | education, socio-economic status, nationality, personal appearance, race, | ||||||
|  | religion, or sexual identity and orientation. | ||||||
|  |  | ||||||
|  | Our Standards | ||||||
|  |  | ||||||
|  | Examples of behavior that contributes to creating a positive environment | ||||||
|  | include: | ||||||
|  |  | ||||||
|  |  | ||||||
|  | * Using welcoming and inclusive language | ||||||
|  | * Being respectful of differing viewpoints and experiences | ||||||
|  | * Gracefully accepting constructive criticism | ||||||
|  | * Focusing on what is best for the community | ||||||
|  | * Showing empathy towards other community members | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Examples of unacceptable behavior by participants include: | ||||||
|  |  | ||||||
|  |  | ||||||
|  | * The use of sexualized language or imagery and unwelcome sexual attention or | ||||||
|  | advances | ||||||
|  | * Trolling, insulting/derogatory comments, and personal or political attacks | ||||||
|  | * Public or private harassment | ||||||
|  | * Publishing others’ private information, such as a physical or electronic | ||||||
|  | address, without explicit permission | ||||||
|  | * Other conduct which could reasonably be considered inappropriate in a | ||||||
|  | professional setting | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Our Responsibilities | ||||||
|  |  | ||||||
|  | Project maintainers are responsible for clarifying the standards of acceptable | ||||||
|  | behavior and are expected to take appropriate and fair corrective action in | ||||||
|  | response to any instances of unacceptable behavior. | ||||||
|  |  | ||||||
|  | Project maintainers have the right and responsibility to remove, edit, or | ||||||
|  | reject comments, commits, code, wiki edits, issues, and other contributions | ||||||
|  | that are not aligned to this Code of Conduct, or to ban temporarily or | ||||||
|  | permanently any contributor for other behaviors that they deem inappropriate, | ||||||
|  | threatening, offensive, or harmful. | ||||||
|  |  | ||||||
|  | Scope | ||||||
|  |  | ||||||
|  | This Code of Conduct applies both within project spaces and in public spaces | ||||||
|  | when an individual is representing the project or its community. Examples of | ||||||
|  | representing a project or community include using an official project e-mail | ||||||
|  | address, posting via an official social media account, or acting as an appointed | ||||||
|  | representative at an online or offline event. Representation of a project may be | ||||||
|  | further defined and clarified by project maintainers. | ||||||
|  |  | ||||||
|  | Enforcement | ||||||
|  |  | ||||||
|  | Instances of abusive, harassing, or otherwise unacceptable behavior may be | ||||||
|  | reported by contacting the project team at carpedm20@gmail.com. All | ||||||
|  | complaints will be reviewed and investigated and will result in a response that | ||||||
|  | is deemed necessary and appropriate to the circumstances. The project team is | ||||||
|  | obligated to maintain confidentiality with regard to the reporter of an incident. | ||||||
|  | Further details of specific enforcement policies may be posted separately. | ||||||
|  |  | ||||||
|  | Project maintainers who do not follow or enforce the Code of Conduct in good | ||||||
|  | faith may face temporary or permanent repercussions as determined by other | ||||||
|  | members of the project’s leadership. | ||||||
|  |  | ||||||
|  | Attribution | ||||||
|  |  | ||||||
|  | This Code of Conduct is adapted from the Contributor Covenant, version 1.4, | ||||||
|  | available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html | ||||||
							
								
								
									
										38
									
								
								CONTRIBUTING.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,38 @@ | |||||||
|  | Contributing to ``fbchat`` | ||||||
|  | ========================== | ||||||
|  |  | ||||||
|  | Thanks for reading this, all contributions are very much welcome! | ||||||
|  |  | ||||||
|  | Please be aware that ``fbchat`` uses `Scemantic Versioning <https://semver.org/>`__ | ||||||
|  | That means that if you're submitting a breaking change, it will probably take a while before it gets considered. | ||||||
|  |  | ||||||
|  | In that case, you can point your PR to the ``2.0.0-dev`` branch, where the API is being properly developed. | ||||||
|  | Otherwise, just point it to ``master``. | ||||||
|  |  | ||||||
|  | Development Environment | ||||||
|  | ----------------------- | ||||||
|  |  | ||||||
|  | You can use `flit` to install the package as a symlink: | ||||||
|  |  | ||||||
|  | .. code-block:: | ||||||
|  |  | ||||||
|  |     $ # *nix: | ||||||
|  |     $ flit install --symlink | ||||||
|  |     $ # Windows: | ||||||
|  |     $ flit install --pth-file | ||||||
|  |  | ||||||
|  | After that, you can ``import`` the module as normal. | ||||||
|  |  | ||||||
|  | Before committing, you should run ``black .`` in the main directory, to format your code. | ||||||
|  |  | ||||||
|  | Testing Environment | ||||||
|  | ------------------- | ||||||
|  |  | ||||||
|  | The tests use `pytest <https://docs.pytest.org/>`__, and to work they need two Facebook accounts, and a group thread between these. | ||||||
|  | To set these up, you should export the following environment variables: | ||||||
|  |  | ||||||
|  | ``client1_email``, ``client1_password``, ``client2_email``, ``client2_password`` and ``group_id`` | ||||||
|  |  | ||||||
|  | If you're not able to do this, consider simply running ``pytest -m offline``. | ||||||
|  |  | ||||||
|  | And if you're adding new functionality, if possible, make sure to create a new test for it. | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| New BSD License | BSD 3-Clause License | ||||||
| 
 | 
 | ||||||
| Copyright (c) 2015, Taehoon Kim | Copyright (c) 2015, Taehoon Kim | ||||||
| All rights reserved. | All rights reserved. | ||||||
| @@ -13,8 +13,9 @@ modification, are permitted provided that the following conditions are met: | |||||||
|   this list of conditions and the following disclaimer in the documentation |   this list of conditions and the following disclaimer in the documentation | ||||||
|   and/or other materials provided with the distribution. |   and/or other materials provided with the distribution. | ||||||
| 
 | 
 | ||||||
| * The names of its contributors may not be used to endorse or promote products | * Neither the name of the copyright holder nor the names of its | ||||||
|   derived from this software without specific prior written permission. |   contributors may be used to endorse or promote products derived from | ||||||
|  |   this software without specific prior written permission. | ||||||
| 
 | 
 | ||||||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||||
| AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| include LICENSE.txt |  | ||||||
| include MANIFEST.in |  | ||||||
| include README.rst |  | ||||||
| include setup.py |  | ||||||
							
								
								
									
										139
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						| @@ -1,119 +1,50 @@ | |||||||
| ====== | ``fbchat``: Facebook Chat (Messenger) for Python | ||||||
| fbchat | ================================================ | ||||||
| ====== |  | ||||||
|  |  | ||||||
|  | .. image:: https://img.shields.io/badge/license-BSD-blue.svg | ||||||
|  |     :target: https://github.com/carpedm20/fbchat/tree/master/LICENSE | ||||||
|  |     :alt: License: BSD 3-Clause | ||||||
|  |  | ||||||
| Facebook Chat (`Messenger <https://www.messenger.com/>`__) for Python. This project was inspired by `facebook-chat-api <https://github.com/Schmavery/facebook-chat-api>`__. | .. image:: https://img.shields.io/badge/python-2.7%2C%203.4%2C%203.5%2C%203.6%203.7%20pypy-blue.svg | ||||||
|  |     :target: https://pypi.python.org/pypi/fbchat | ||||||
|  |     :alt: Supported python versions: 2.7, 3.4, 3.5, 3.6, 3.7 and pypy | ||||||
|  |  | ||||||
| **No XMPP or API key is needed**. Just use your ID and PASSWORD. | .. image:: https://readthedocs.org/projects/fbchat/badge/?version=latest | ||||||
|  |     :target: https://fbchat.readthedocs.io | ||||||
|  |     :alt: Documentation | ||||||
|  |  | ||||||
|  | .. image:: https://travis-ci.org/carpedm20/fbchat.svg?branch=master | ||||||
|  |     :target: https://travis-ci.org/carpedm20/fbchat | ||||||
|  |     :alt: Travis CI | ||||||
|  |  | ||||||
| Installation | .. image:: https://img.shields.io/badge/code%20style-black-000000.svg | ||||||
| ============ |     :target: https://github.com/ambv/black | ||||||
|  |     :alt: Code style | ||||||
|  |  | ||||||
| Simple: | Facebook Chat (`Messenger <https://www.facebook.com/messages/>`__) for Python. | ||||||
|  | This project was inspired by `facebook-chat-api <https://github.com/Schmavery/facebook-chat-api>`__. | ||||||
|  |  | ||||||
| .. code-block:: console | **No XMPP or API key is needed**. Just use your email and password. | ||||||
|  |  | ||||||
|  | Go to `Read the Docs <https://fbchat.readthedocs.io>`__ to see the full documentation, | ||||||
|  | or jump right into the code by viewing the `examples <https://github.com/carpedm20/fbchat/tree/master/examples>`__ | ||||||
|  |  | ||||||
|  | Installation: | ||||||
|  |  | ||||||
|  | .. code-block:: | ||||||
|  |  | ||||||
|     $ pip install fbchat |     $ pip install fbchat | ||||||
|  |  | ||||||
|  | You can also install from source if you have ``pip>=19.0``: | ||||||
|  |  | ||||||
| Example | .. code-block:: | ||||||
| ======= |  | ||||||
|  |  | ||||||
| .. code-block:: python |     $ git clone https://github.com/carpedm20/fbchat.git | ||||||
|  |     $ pip install fbchat | ||||||
|     import fbchat |  | ||||||
|  |  | ||||||
|     client = fbchat.Client("YOUR_ID", "YOUR_PASSWORD") |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Sending a Message | Maintainer | ||||||
| ================= | ---------- | ||||||
|  |  | ||||||
| .. code-block:: python | - Mads Marquart / `@madsmtm <https://github.com/madsmtm>`__ | ||||||
|      | - Taehoon Kim / `@carpedm20 <http://carpedm20.github.io/about/>`__ | ||||||
|     friends = client.getUsers("FRIEND'S NAME")  # return a list of names |  | ||||||
|     friend = friends[0] |  | ||||||
|     sent = client.send(friend.uid, "Your Message") |  | ||||||
|     if sent: |  | ||||||
|         print("Message sent successfully!") |  | ||||||
|     # IMAGES |  | ||||||
|     client.sendLocalImage(friend.uid,message='<message text>',image='<path/to/image/file>') # send local image |  | ||||||
|     imgurl = "http://i.imgur.com/LDQ2ITV.jpg" |  | ||||||
|     client.sendRemoteImage(friend.uid,message='<message text>', image=imgurl) # send image from image url |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Getting user info from user id |  | ||||||
| ============================== |  | ||||||
|  |  | ||||||
| .. code-block:: python |  | ||||||
|  |  | ||||||
|     friend1 = client.getUsers('<friend name 1>')[0] |  | ||||||
|     friend2 = client.getUsers('<friend name 2>')[0] |  | ||||||
|     friend1_info = client.getUserInfo(friend1.uid) # returns dict with details |  | ||||||
|     both_info = client.getUserInfo(friend1.uid,friend2.uid) # query both together, returns list of dicts |  | ||||||
|     friend1_name = friend1_info['name']  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Getting last messages sent |  | ||||||
| ========================== |  | ||||||
|  |  | ||||||
| .. code-block:: python |  | ||||||
|      |  | ||||||
|     last_messages = client.getThreadInfo(friend.uid, last_n=20) |  | ||||||
|     last_messages.reverse()  # messages come in reversed order |  | ||||||
|      |  | ||||||
|     for message in last_messages: |  | ||||||
|         print(message.body) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Example Echobot |  | ||||||
| =============== |  | ||||||
|  |  | ||||||
| .. code-block:: python |  | ||||||
|  |  | ||||||
|     import fbchat |  | ||||||
|     #subclass fbchat.Client and override required methods |  | ||||||
|     class EchoBot(fbchat.Client):  |  | ||||||
|  |  | ||||||
|         def __init__(self,email, password, debug=True, user_agent=None):             |  | ||||||
|             fbchat.Client.__init__(self,email, password, debug, user_agent) |  | ||||||
|  |  | ||||||
|         def on_message(self, mid, author_id, author_name, message, metadata): |  | ||||||
|             self.markAsDelivered(author_id, mid) #mark delivered |  | ||||||
|             self.markAsRead(author_id) #mark read |  | ||||||
|  |  | ||||||
|             print("%s said: %s"%(author_id, message)) |  | ||||||
|  |  | ||||||
|             #if you are not the author, echo |  | ||||||
|             if str(author_id) != str(self.uid): |  | ||||||
|                 self.send(author_id,message) |  | ||||||
|      |  | ||||||
|     bot = EchoBot("<email>", "<password>") |  | ||||||
|     bot.listen() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Saving session |  | ||||||
| ========================== |  | ||||||
|  |  | ||||||
| .. code-block:: python |  | ||||||
|      |  | ||||||
|     session_cookies = client.setSession() |  | ||||||
|     # save session_cookies |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Loading session |  | ||||||
| ========================== |  | ||||||
|  |  | ||||||
| .. code-block:: python |  | ||||||
|      |  | ||||||
|     client = fbchat.Client(None, None, session_cookies=session_cookies) |  | ||||||
|     # OR |  | ||||||
|     client.setSession(session_cookies) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Authors |  | ||||||
| ======= |  | ||||||
|  |  | ||||||
| Taehoon Kim / `@carpedm20 <http://carpedm20.github.io/about/>`__ |  | ||||||
|   | |||||||
							
								
								
									
										19
									
								
								docs/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | |||||||
|  | # Minimal makefile for Sphinx documentation | ||||||
|  | # | ||||||
|  |  | ||||||
|  | # You can set these variables from the command line. | ||||||
|  | SPHINXOPTS    = | ||||||
|  | SPHINXBUILD   = sphinx-build | ||||||
|  | SOURCEDIR     = . | ||||||
|  | BUILDDIR      = _build | ||||||
|  |  | ||||||
|  | # Put it first so that "make" without argument is like "make help". | ||||||
|  | help: | ||||||
|  | 	@$(SPHINXBUILD) -M help "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O) | ||||||
|  |  | ||||||
|  | .PHONY: help Makefile | ||||||
|  |  | ||||||
|  | # Catch-all target: route all unknown targets to Sphinx using the new | ||||||
|  | # "make mode" option.  $(O) is meant as a shortcut for $(SPHINXOPTS). | ||||||
|  | %: Makefile | ||||||
|  | 	@$(SPHINXBUILD) -M $@ "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O) | ||||||
							
								
								
									
										
											BIN
										
									
								
								docs/_static/find-group-id.png
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 54 KiB | 
							
								
								
									
										1
									
								
								docs/_static/license.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="80" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="80" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h47v20H0z"/><path fill="#007ec6" d="M47 0h33v20H47z"/><path fill="url(#b)" d="M0 0h80v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="11"><text x="23.5" y="15" fill="#010101" fill-opacity=".3">license</text><text x="23.5" y="14">license</text><text x="62.5" y="15" fill="#010101" fill-opacity=".3">BSD</text><text x="62.5" y="14">BSD</text></g></svg> | ||||||
| After Width: | Height: | Size: 791 B | 
							
								
								
									
										1
									
								
								docs/_static/python-versions.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="154" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="154" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h49v20H0z"/><path fill="#007ec6" d="M49 0h105v20H49z"/><path fill="url(#b)" d="M0 0h154v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="11"><text x="24.5" y="15" fill="#010101" fill-opacity=".3">python</text><text x="24.5" y="14">python</text><text x="100.5" y="15" fill="#010101" fill-opacity=".3">2.7, 3.4, 3.5, 3.6</text><text x="100.5" y="14">2.7, 3.4, 3.5, 3.6</text></g></svg> | ||||||
| After Width: | Height: | Size: 825 B | 
							
								
								
									
										26
									
								
								docs/_templates/layout.html
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | |||||||
|  | {% extends '!layout.html' %} | ||||||
|  |  | ||||||
|  | {% block extrahead %} | ||||||
|  | <script async defer src="https://buttons.github.io/buttons.js"></script> | ||||||
|  | <!-- Alabaster (krTheme++) Hacks, modified version of Kenneth Reitz' https://github.com/kennethreitz/requests/blob/master/docs/_templates/hacks.html --> | ||||||
|  | <style type="text/css"> | ||||||
|  |     /* Rezzy requires precise alignment. */ | ||||||
|  |     img.logo {margin-left: -20px!important;} | ||||||
|  |     /* "Quick Search" should be capitalized. */ | ||||||
|  |     div#searchbox h3 {text-transform: capitalize;} | ||||||
|  |     /* Go button should be behind input field */ | ||||||
|  |     div.sphinxsidebar div#searchbox input[type="text"] {width: 160px} | ||||||
|  |     div#searchbox form div {display: inline-block;} | ||||||
|  |     /* Make the document a little wider, less code is cut-off. */ | ||||||
|  |     div.document {width: 1008px;} | ||||||
|  |     /* Much-improved spacing around code blocks. */ | ||||||
|  |     div.highlight pre {padding: 11px 14px;} | ||||||
|  |     /* Remain Responsive! */ | ||||||
|  |     @media screen and (max-width: 1008px) { | ||||||
|  |         div.sphinxsidebar {display: none;} | ||||||
|  |         div.document {width: 100%!important;} | ||||||
|  |         /* Have code blocks escape the document right-margin. */ | ||||||
|  |         div.highlight pre {margin-right: -30px;} | ||||||
|  |     } | ||||||
|  | </style> | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										13
									
								
								docs/_templates/sidebar.html
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,13 @@ | |||||||
|  | <h3> | ||||||
|  |     <a href="{{ pathto(master_doc) }}">{{ _(project) }}</a> | ||||||
|  | </h3> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  |     <a class="github-button" href="https://github.com/carpedm20/fbchat" data-size="large" data-show-count="true" aria-label="Star carpedm20/fbchat on GitHub">Star</a> | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  |     {{ _(shorttitle) }} | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | {{ toctree() }} | ||||||
							
								
								
									
										79
									
								
								docs/api.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,79 @@ | |||||||
|  | .. module:: fbchat | ||||||
|  | .. _api: | ||||||
|  |  | ||||||
|  | .. Note: we're using () to hide the __init__ method where relevant | ||||||
|  |  | ||||||
|  | Full API | ||||||
|  | ======== | ||||||
|  |  | ||||||
|  | If you are looking for information on a specific function, class, or method, this part of the documentation is for you. | ||||||
|  |  | ||||||
|  | Client | ||||||
|  | ------ | ||||||
|  |  | ||||||
|  | .. autoclass:: Client | ||||||
|  |  | ||||||
|  | Threads | ||||||
|  | ------- | ||||||
|  |  | ||||||
|  | .. autoclass:: Thread() | ||||||
|  | .. autoclass:: ThreadType(Enum) | ||||||
|  |     :undoc-members: | ||||||
|  | .. autoclass:: Page() | ||||||
|  | .. autoclass:: User() | ||||||
|  | .. autoclass:: Group() | ||||||
|  |  | ||||||
|  | Messages | ||||||
|  | -------- | ||||||
|  |  | ||||||
|  | .. autoclass:: Message | ||||||
|  | .. autoclass:: Mention | ||||||
|  | .. autoclass:: EmojiSize(Enum) | ||||||
|  |     :undoc-members: | ||||||
|  | .. autoclass:: MessageReaction(Enum) | ||||||
|  |     :undoc-members: | ||||||
|  |  | ||||||
|  | Exceptions | ||||||
|  | ---------- | ||||||
|  |  | ||||||
|  | .. autoexception:: FBchatException() | ||||||
|  | .. autoexception:: FBchatFacebookError() | ||||||
|  | .. autoexception:: FBchatUserError() | ||||||
|  |  | ||||||
|  | Attachments | ||||||
|  | ----------- | ||||||
|  |  | ||||||
|  | .. autoclass:: Attachment() | ||||||
|  | .. autoclass:: ShareAttachment() | ||||||
|  | .. autoclass:: Sticker() | ||||||
|  | .. autoclass:: LocationAttachment() | ||||||
|  | .. autoclass:: LiveLocationAttachment() | ||||||
|  | .. autoclass:: FileAttachment() | ||||||
|  | .. autoclass:: AudioAttachment() | ||||||
|  | .. autoclass:: ImageAttachment() | ||||||
|  | .. autoclass:: VideoAttachment() | ||||||
|  | .. autoclass:: ImageAttachment() | ||||||
|  |  | ||||||
|  | Miscellaneous | ||||||
|  | ------------- | ||||||
|  |  | ||||||
|  | .. autoclass:: ThreadLocation(Enum) | ||||||
|  |     :undoc-members: | ||||||
|  | .. autoclass:: ThreadColor(Enum) | ||||||
|  |     :undoc-members: | ||||||
|  | .. autoclass:: ActiveStatus() | ||||||
|  | .. autoclass:: TypingStatus(Enum) | ||||||
|  |     :undoc-members: | ||||||
|  |  | ||||||
|  | .. autoclass:: QuickReply | ||||||
|  | .. autoclass:: QuickReplyText | ||||||
|  | .. autoclass:: QuickReplyLocation | ||||||
|  | .. autoclass:: QuickReplyPhoneNumber | ||||||
|  | .. autoclass:: QuickReplyEmail | ||||||
|  |  | ||||||
|  | .. autoclass:: Poll | ||||||
|  | .. autoclass:: PollOption | ||||||
|  |  | ||||||
|  | .. autoclass:: Plan | ||||||
|  | .. autoclass:: GuestStatus(Enum) | ||||||
|  |     :undoc-members: | ||||||
							
								
								
									
										208
									
								
								docs/conf.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,208 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | # | ||||||
|  | # Configuration file for the Sphinx documentation builder. | ||||||
|  | # | ||||||
|  | # This file does only contain a selection of the most common options. For a | ||||||
|  | # full list see the documentation: | ||||||
|  | # http://www.sphinx-doc.org/en/master/config | ||||||
|  |  | ||||||
|  | # -- Path setup -------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | import os | ||||||
|  | import sys | ||||||
|  |  | ||||||
|  | sys.path.insert(0, os.path.abspath("..")) | ||||||
|  |  | ||||||
|  | import fbchat | ||||||
|  |  | ||||||
|  | # -- Project information ----------------------------------------------------- | ||||||
|  |  | ||||||
|  | project = fbchat.__name__ | ||||||
|  | copyright = fbchat.__copyright__ | ||||||
|  | author = fbchat.__author__ | ||||||
|  |  | ||||||
|  | # The short X.Y version | ||||||
|  | version = fbchat.__version__ | ||||||
|  | # The full version, including alpha/beta/rc tags | ||||||
|  | release = fbchat.__version__ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- General configuration --------------------------------------------------- | ||||||
|  |  | ||||||
|  | # If your documentation needs a minimal Sphinx version, state it here. | ||||||
|  | # | ||||||
|  | needs_sphinx = "2.0" | ||||||
|  |  | ||||||
|  | # 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", | ||||||
|  |     "sphinx.ext.intersphinx", | ||||||
|  |     "sphinx.ext.todo", | ||||||
|  |     "sphinx.ext.viewcode", | ||||||
|  |     "sphinx.ext.napoleon", | ||||||
|  |     "sphinxcontrib.spelling", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | # Add any paths that contain templates here, relative to this directory. | ||||||
|  | templates_path = ["_templates"] | ||||||
|  |  | ||||||
|  | # The master toctree document. | ||||||
|  | master_doc = "index" | ||||||
|  |  | ||||||
|  | # List of patterns, relative to source directory, that match files and | ||||||
|  | # directories to ignore when looking for source files. | ||||||
|  | # This pattern also affects html_static_path and html_extra_path. | ||||||
|  | exclude_patterns = ["_build", "Thumbs.db", ".DS_Store"] | ||||||
|  |  | ||||||
|  | rst_prolog = ".. currentmodule:: " + project | ||||||
|  |  | ||||||
|  | # The reST default role (used for this markup: `text`) to use for all | ||||||
|  | # documents. | ||||||
|  | # | ||||||
|  | default_role = "any" | ||||||
|  |  | ||||||
|  | # Make the reference parsing more strict | ||||||
|  | # | ||||||
|  | nitpicky = True | ||||||
|  |  | ||||||
|  | # Prefer strict Python highlighting | ||||||
|  | # | ||||||
|  | highlight_language = "python3" | ||||||
|  |  | ||||||
|  | # If true, '()' will be appended to :func: etc. cross-reference text. | ||||||
|  | # | ||||||
|  | add_function_parentheses = False | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- 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 = "alabaster" | ||||||
|  |  | ||||||
|  | # Theme options are theme-specific and customize the look and feel of a theme | ||||||
|  | # further.  For a list of options available for each theme, see the | ||||||
|  | # documentation. | ||||||
|  | # | ||||||
|  | html_theme_options = { | ||||||
|  |     "show_powered_by": False, | ||||||
|  |     "github_user": "carpedm20", | ||||||
|  |     "github_repo": project, | ||||||
|  |     "github_banner": True, | ||||||
|  |     "show_related": False, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Custom sidebar templates, must be a dictionary that maps document names | ||||||
|  | # to template names. | ||||||
|  | # | ||||||
|  | # The default sidebars (for documents that don't match any pattern) are | ||||||
|  | # defined by theme itself.  Builtin themes are using these templates by | ||||||
|  | # default: ``['localtoc.html', 'relations.html', 'sourcelink.html', | ||||||
|  | # 'searchbox.html']``. | ||||||
|  | # | ||||||
|  | html_sidebars = {"**": ["sidebar.html", "searchbox.html"]} | ||||||
|  |  | ||||||
|  | # If true, "Created using Sphinx" is shown in the HTML footer. Default is True. | ||||||
|  | # | ||||||
|  | html_show_sphinx = False | ||||||
|  |  | ||||||
|  | # If true, links to the reST sources are added to the pages. | ||||||
|  | # | ||||||
|  | html_show_sourcelink = False | ||||||
|  |  | ||||||
|  | # A shorter title for the navigation bar. Default is the same as html_title. | ||||||
|  | # | ||||||
|  | html_short_title = fbchat.__description__ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- Options for HTMLHelp output --------------------------------------------- | ||||||
|  |  | ||||||
|  | # Output file base name for HTML help builder. | ||||||
|  | htmlhelp_basename = project + "doc" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- Options for LaTeX output ------------------------------------------------ | ||||||
|  |  | ||||||
|  | # Grouping the document tree into LaTeX files. List of tuples | ||||||
|  | # (source start file, target name, title, | ||||||
|  | #  author, documentclass [howto, manual, or own class]). | ||||||
|  | latex_documents = [(master_doc, project + ".tex", fbchat.__title__, author, "manual")] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- Options for manual page output ------------------------------------------ | ||||||
|  |  | ||||||
|  | # One entry per manual page. List of tuples | ||||||
|  | # (source start file, name, description, authors, manual section). | ||||||
|  | man_pages = [ | ||||||
|  |     (master_doc, project, fbchat.__title__, [x.strip() for x in author.split(";")], 1) | ||||||
|  | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- Options for Texinfo output ---------------------------------------------- | ||||||
|  |  | ||||||
|  | # Grouping the document tree into Texinfo files. List of tuples | ||||||
|  | # (source start file, target name, title, author, | ||||||
|  | #  dir menu entry, description, category) | ||||||
|  | texinfo_documents = [ | ||||||
|  |     ( | ||||||
|  |         master_doc, | ||||||
|  |         project, | ||||||
|  |         fbchat.__title__, | ||||||
|  |         author, | ||||||
|  |         project, | ||||||
|  |         fbchat.__description__, | ||||||
|  |         "Miscellaneous", | ||||||
|  |     ) | ||||||
|  | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- Options for Epub output ------------------------------------------------- | ||||||
|  |  | ||||||
|  | # A list of files that should not be packed into the epub file. | ||||||
|  | epub_exclude_files = ["search.html"] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- Extension configuration ------------------------------------------------- | ||||||
|  |  | ||||||
|  | # -- Options for autodoc extension --------------------------------------- | ||||||
|  |  | ||||||
|  | autoclass_content = "both" | ||||||
|  | autodoc_member_order = "bysource" | ||||||
|  | autodoc_default_options = {"members": True} | ||||||
|  |  | ||||||
|  | # -- Options for intersphinx extension --------------------------------------- | ||||||
|  |  | ||||||
|  | # Example configuration for intersphinx: refer to the Python standard library. | ||||||
|  | intersphinx_mapping = {"https://docs.python.org/": None} | ||||||
|  |  | ||||||
|  | # -- Options for todo extension ---------------------------------------------- | ||||||
|  |  | ||||||
|  | # If true, `todo` and `todoList` produce output, else they produce nothing. | ||||||
|  | todo_include_todos = True | ||||||
|  |  | ||||||
|  | todo_link_only = True | ||||||
|  |  | ||||||
|  | # -- Options for napoleon extension ---------------------------------------------- | ||||||
|  |  | ||||||
|  | # Use Google style docstrings | ||||||
|  | napoleon_google_docstring = True | ||||||
|  | napoleon_numpy_docstring = False | ||||||
|  |  | ||||||
|  | # napoleon_use_admonition_for_examples = False | ||||||
|  | # napoleon_use_admonition_for_notes = False | ||||||
|  | # napoleon_use_admonition_for_references = False | ||||||
|  |  | ||||||
|  | # -- Options for spelling extension ---------------------------------------------- | ||||||
|  |  | ||||||
|  | spelling_word_list_filename = [ | ||||||
|  |     "spelling/names.txt", | ||||||
|  |     "spelling/technical.txt", | ||||||
|  |     "spelling/fixes.txt", | ||||||
|  | ] | ||||||
|  | spelling_ignore_wiki_words = False | ||||||
|  | # spelling_ignore_acronyms = False | ||||||
|  | spelling_ignore_python_builtins = False | ||||||
|  | spelling_ignore_importable_modules = False | ||||||
							
								
								
									
										55
									
								
								docs/examples.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,55 @@ | |||||||
|  | .. _examples: | ||||||
|  |  | ||||||
|  | Examples | ||||||
|  | ======== | ||||||
|  |  | ||||||
|  | These are a few examples on how to use ``fbchat``. Remember to swap out ``<email>`` and ``<password>`` for your email and password | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Basic example | ||||||
|  | ------------- | ||||||
|  |  | ||||||
|  | This will show basic usage of ``fbchat`` | ||||||
|  |  | ||||||
|  | .. literalinclude:: ../examples/basic_usage.py | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Interacting with Threads | ||||||
|  | ------------------------ | ||||||
|  |  | ||||||
|  | This will interact with the thread in every way ``fbchat`` supports | ||||||
|  |  | ||||||
|  | .. literalinclude:: ../examples/interract.py | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Fetching Information | ||||||
|  | -------------------- | ||||||
|  |  | ||||||
|  | This will show the different ways of fetching information about users and threads | ||||||
|  |  | ||||||
|  | .. literalinclude:: ../examples/fetch.py | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ``Echobot`` | ||||||
|  | ----------- | ||||||
|  |  | ||||||
|  | This will reply to any message with the same message | ||||||
|  |  | ||||||
|  | .. literalinclude:: ../examples/echobot.py | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Remove Bot | ||||||
|  | ---------- | ||||||
|  |  | ||||||
|  | This will remove a user from a group if they write the message ``Remove me!`` | ||||||
|  |  | ||||||
|  | .. literalinclude:: ../examples/removebot.py | ||||||
|  |  | ||||||
|  |  | ||||||
|  | "Prevent changes"-Bot | ||||||
|  | --------------------- | ||||||
|  |  | ||||||
|  | This will prevent chat color, emoji, nicknames and chat name from being changed. | ||||||
|  | It will also prevent people from being added and removed | ||||||
|  |  | ||||||
|  | .. literalinclude:: ../examples/keepbot.py | ||||||
							
								
								
									
										42
									
								
								docs/faq.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,42 @@ | |||||||
|  | .. _faq: | ||||||
|  |  | ||||||
|  | FAQ | ||||||
|  | === | ||||||
|  |  | ||||||
|  | Version X broke my installation | ||||||
|  | ------------------------------- | ||||||
|  |  | ||||||
|  | We try to provide backwards compatibility where possible, but since we're not part of Facebook, | ||||||
|  | most of the things may be broken at any point in time | ||||||
|  |  | ||||||
|  | Downgrade to an earlier version of ``fbchat``, run this command | ||||||
|  |  | ||||||
|  | .. code-block:: sh | ||||||
|  |  | ||||||
|  |     $ pip install fbchat==<X> | ||||||
|  |  | ||||||
|  | Where you replace ``<X>`` with the version you want to use | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Will you be supporting creating posts/events/pages and so on? | ||||||
|  | ------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | We won't be focusing on anything else than chat-related things. This API is called ``fbCHAT``, after all ;) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Submitting Issues | ||||||
|  | ----------------- | ||||||
|  |  | ||||||
|  | If you're having trouble with some of the snippets, or you think some of the functionality is broken, | ||||||
|  | please feel free to submit an issue on `GitHub <https://github.com/carpedm20/fbchat>`_. | ||||||
|  | You should first login with ``logging_level`` set to ``logging.DEBUG``:: | ||||||
|  |  | ||||||
|  |     from fbchat import Client | ||||||
|  |     import logging | ||||||
|  |     client = Client('<email>', '<password>', logging_level=logging.DEBUG) | ||||||
|  |  | ||||||
|  | Then you can submit the relevant parts of this log, and detailed steps on how to reproduce | ||||||
|  |  | ||||||
|  | .. warning:: | ||||||
|  |     Always remove your credentials from any debug information you may provide us. | ||||||
|  |     Preferably, use a test account, in case you miss anything | ||||||
							
								
								
									
										64
									
								
								docs/index.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,64 @@ | |||||||
|  | .. fbchat documentation master file, created by | ||||||
|  |    sphinx-quickstart on Thu May 25 15:43:01 2017. | ||||||
|  |    You can adapt this file completely to your liking, but it should at least | ||||||
|  |    contain the root `toctree` directive. | ||||||
|  |  | ||||||
|  | .. This documentation's layout is heavily inspired by requests' layout: https://requests.readthedocs.io | ||||||
|  |    Some documentation is also partially copied from facebook-chat-api: https://github.com/Schmavery/facebook-chat-api | ||||||
|  |  | ||||||
|  | ``fbchat``: Facebook Chat (Messenger) for Python | ||||||
|  | ================================================ | ||||||
|  |  | ||||||
|  | Release v\ |version|. (:ref:`install`) | ||||||
|  |  | ||||||
|  | .. generated with: https://img.shields.io/badge/license-BSD-blue.svg | ||||||
|  |  | ||||||
|  | .. image:: /_static/license.svg | ||||||
|  |     :target: https://github.com/carpedm20/fbchat/blob/master/LICENSE.txt | ||||||
|  |     :alt: License: BSD | ||||||
|  |  | ||||||
|  | .. generated with: https://img.shields.io/badge/python-2.7%2C%203.4%2C%203.5%2C%203.6-blue.svg | ||||||
|  |  | ||||||
|  | .. image:: /_static/python-versions.svg | ||||||
|  |     :target: https://pypi.python.org/pypi/fbchat | ||||||
|  |     :alt: Supported python versions: 2.7, 3.4, 3.5 and 3.6 | ||||||
|  |  | ||||||
|  | Facebook Chat (`Messenger <https://www.facebook.com/messages/>`_) for Python. | ||||||
|  | This project was inspired by `facebook-chat-api <https://github.com/Schmavery/facebook-chat-api>`_. | ||||||
|  |  | ||||||
|  | **No XMPP or API key is needed**. Just use your email and password. | ||||||
|  |  | ||||||
|  | Currently ``fbchat`` support Python 2.7, 3.4, 3.5 and 3.6: | ||||||
|  |  | ||||||
|  | ``fbchat`` works by emulating the browser. | ||||||
|  | This means doing the exact same GET/POST requests and tricking Facebook into thinking it's accessing the website normally. | ||||||
|  | Therefore, this API requires the credentials of a Facebook account. | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |     If you're having problems, please check the :ref:`faq`, before asking questions on GitHub | ||||||
|  |  | ||||||
|  | .. warning:: | ||||||
|  |     We are not responsible if your account gets banned for spammy activities, | ||||||
|  |     such as sending lots of messages to people you don't know, sending messages very quickly, | ||||||
|  |     sending spammy looking URLs, logging in and out very quickly... Be responsible Facebook citizens. | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |     Facebook now has an `official API <https://developers.facebook.com/docs/messenger-platform>`_ for chat bots, | ||||||
|  |     so if you're familiar with ``Node.js``, this might be what you're looking for. | ||||||
|  |  | ||||||
|  | If you're already familiar with the basics of how Facebook works internally, go to :ref:`examples` to see example usage of ``fbchat`` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Overview | ||||||
|  | -------- | ||||||
|  |  | ||||||
|  | .. toctree:: | ||||||
|  |     :maxdepth: 2 | ||||||
|  |  | ||||||
|  |     install | ||||||
|  |     intro | ||||||
|  |     examples | ||||||
|  |     testing | ||||||
|  |     api | ||||||
|  |     todo | ||||||
|  |     faq | ||||||
							
								
								
									
										43
									
								
								docs/install.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,43 @@ | |||||||
|  | .. _install: | ||||||
|  |  | ||||||
|  | Installation | ||||||
|  | ============ | ||||||
|  |  | ||||||
|  | Install using pip | ||||||
|  | ----------------- | ||||||
|  |  | ||||||
|  | To install ``fbchat``, run this command: | ||||||
|  |  | ||||||
|  | .. code-block:: sh | ||||||
|  |  | ||||||
|  |     $ pip install fbchat | ||||||
|  |  | ||||||
|  | If you don't have `pip <https://pip.pypa.io>`_ installed, | ||||||
|  | `this Python installation guide <http://docs.python-guide.org/en/latest/starting/installation/>`_ | ||||||
|  | can guide you through the process. | ||||||
|  |  | ||||||
|  | Get the Source Code | ||||||
|  | ------------------- | ||||||
|  |  | ||||||
|  | ``fbchat`` is developed on GitHub, where the code is | ||||||
|  | `always available <https://github.com/carpedm20/fbchat>`_. | ||||||
|  |  | ||||||
|  | You can either clone the public repository: | ||||||
|  |  | ||||||
|  | .. code-block:: sh | ||||||
|  |  | ||||||
|  |     $ git clone git://github.com/carpedm20/fbchat.git | ||||||
|  |  | ||||||
|  | Or, download a `tarball <https://github.com/carpedm20/fbchat/tarball/master>`_: | ||||||
|  |  | ||||||
|  | .. code-block:: sh | ||||||
|  |  | ||||||
|  |     $ curl -OL https://github.com/carpedm20/fbchat/tarball/master | ||||||
|  |     # optionally, zipball is also available (for Windows users). | ||||||
|  |  | ||||||
|  | Once you have a copy of the source, you can embed it in your own Python | ||||||
|  | package, or install it into your site-packages easily: | ||||||
|  |  | ||||||
|  | .. code-block:: sh | ||||||
|  |  | ||||||
|  |     $ python setup.py install | ||||||
							
								
								
									
										198
									
								
								docs/intro.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,198 @@ | |||||||
|  | .. _intro: | ||||||
|  |  | ||||||
|  | Introduction | ||||||
|  | ============ | ||||||
|  |  | ||||||
|  | ``fbchat`` uses your email and password to communicate with the Facebook server. | ||||||
|  | That means that you should always store your password in a separate file, in case e.g. someone looks over your shoulder while you're writing code. | ||||||
|  | You should also make sure that the file's access control is appropriately restrictive | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. _intro_logging_in: | ||||||
|  |  | ||||||
|  | Logging In | ||||||
|  | ---------- | ||||||
|  |  | ||||||
|  | Simply create an instance of :class:`Client`. If you have two factor authentication enabled, type the code in the terminal prompt | ||||||
|  | (If you want to supply the code in another fashion, overwrite :func:`Client.on2FACode`):: | ||||||
|  |  | ||||||
|  |     from fbchat import Client | ||||||
|  |     from fbchat.models import * | ||||||
|  |     client = Client('<email>', '<password>') | ||||||
|  |  | ||||||
|  | Replace ``<email>`` and ``<password>`` with your email and password respectively | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |     For ease of use then most of the code snippets in this document will assume you've already completed the login process | ||||||
|  |     Though the second line, ``from fbchat.models import *``, is not strictly necessary here, later code snippets will assume you've done this | ||||||
|  |  | ||||||
|  | If you want to change how verbose ``fbchat`` is, change the logging level (in :class:`Client`) | ||||||
|  |  | ||||||
|  | Throughout your code, if you want to check whether you are still logged in, use :func:`Client.isLoggedIn`. | ||||||
|  | An example would be to login again if you've been logged out, using :func:`Client.login`:: | ||||||
|  |  | ||||||
|  |     if not client.isLoggedIn(): | ||||||
|  |         client.login('<email>', '<password>') | ||||||
|  |  | ||||||
|  | When you're done using the client, and want to securely logout, use :func:`Client.logout`:: | ||||||
|  |  | ||||||
|  |     client.logout() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. _intro_threads: | ||||||
|  |  | ||||||
|  | Threads | ||||||
|  | ------- | ||||||
|  |  | ||||||
|  | A thread can refer to two things: A Messenger group chat or a single Facebook user | ||||||
|  |  | ||||||
|  | :class:`ThreadType` is an enumerator with two values: ``USER`` and ``GROUP``. | ||||||
|  | These will specify whether the thread is a single user chat or a group chat. | ||||||
|  | This is required for many of ``fbchat``'s functions, since Facebook differentiates between these two internally | ||||||
|  |  | ||||||
|  | Searching for group chats and finding their ID can be done via. :func:`Client.searchForGroups`, | ||||||
|  | and searching for users is possible via. :func:`Client.searchForUsers`. See :ref:`intro_fetching` | ||||||
|  |  | ||||||
|  | You can get your own user ID by using :any:`Client.uid` | ||||||
|  |  | ||||||
|  | Getting the ID of a group chat is fairly trivial otherwise, since you only need to navigate to `<https://www.facebook.com/messages/>`_, | ||||||
|  | click on the group you want to find the ID of, and then read the id from the address bar. | ||||||
|  | The URL will look something like this: ``https://www.facebook.com/messages/t/1234567890``, where ``1234567890`` would be the ID of the group. | ||||||
|  | An image to illustrate this is shown below: | ||||||
|  |  | ||||||
|  | .. image:: /_static/find-group-id.png | ||||||
|  |     :alt: An image illustrating how to find the ID of a group | ||||||
|  |  | ||||||
|  | The same method can be applied to some user accounts, though if they've set a custom URL, then you'll just see that URL instead | ||||||
|  |  | ||||||
|  | Here's an snippet showing the usage of thread IDs and thread types, where ``<user id>`` and ``<group id>`` | ||||||
|  | corresponds to the ID of a single user, and the ID of a group respectively:: | ||||||
|  |  | ||||||
|  |     client.send(Message(text='<message>'), thread_id='<user id>', thread_type=ThreadType.USER) | ||||||
|  |     client.send(Message(text='<message>'), thread_id='<group id>', thread_type=ThreadType.GROUP) | ||||||
|  |  | ||||||
|  | Some functions (e.g. :func:`Client.changeThreadColor`) don't require a thread type, so in these cases you just provide the thread ID:: | ||||||
|  |  | ||||||
|  |     client.changeThreadColor(ThreadColor.BILOBA_FLOWER, thread_id='<user id>') | ||||||
|  |     client.changeThreadColor(ThreadColor.MESSENGER_BLUE, thread_id='<group id>') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. _intro_message_ids: | ||||||
|  |  | ||||||
|  | Message IDs | ||||||
|  | ----------- | ||||||
|  |  | ||||||
|  | Every message you send on Facebook has a unique ID, and every action you do in a thread, | ||||||
|  | like changing a nickname or adding a person, has a unique ID too. | ||||||
|  |  | ||||||
|  | Some of ``fbchat``'s functions require these ID's, like :func:`Client.reactToMessage`, | ||||||
|  | and some of then provide this ID, like :func:`Client.sendMessage`. | ||||||
|  | This snippet shows how to send a message, and then use the returned ID to react to that message with a 😍 emoji:: | ||||||
|  |  | ||||||
|  |     message_id = client.send(Message(text='message'), thread_id=thread_id, thread_type=thread_type) | ||||||
|  |     client.reactToMessage(message_id, MessageReaction.LOVE) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. _intro_interacting: | ||||||
|  |  | ||||||
|  | Interacting with Threads | ||||||
|  | ------------------------ | ||||||
|  |  | ||||||
|  | ``fbchat`` provides multiple functions for interacting with threads | ||||||
|  |  | ||||||
|  | Most functionality works on all threads, though some things, | ||||||
|  | like adding users to and removing users from a group chat, logically only works on group chats | ||||||
|  |  | ||||||
|  | The simplest way of using ``fbchat`` is to send a message. | ||||||
|  | The following snippet will, as you've probably already figured out, send the message ``test message`` to your account:: | ||||||
|  |  | ||||||
|  |     message_id = client.send(Message(text='test message'), thread_id=client.uid, thread_type=ThreadType.USER) | ||||||
|  |  | ||||||
|  | You can see a full example showing all the possible thread interactions with ``fbchat`` by going to :ref:`examples` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. _intro_fetching: | ||||||
|  |  | ||||||
|  | Fetching Information | ||||||
|  | -------------------- | ||||||
|  |  | ||||||
|  | You can use ``fbchat`` to fetch basic information like user names, profile pictures, thread names and user IDs | ||||||
|  |  | ||||||
|  | You can retrieve a user's ID with :func:`Client.searchForUsers`. | ||||||
|  | The following snippet will search for users by their name, take the first (and most likely) user, and then get their user ID from the result:: | ||||||
|  |  | ||||||
|  |     users = client.searchForUsers('<name of user>') | ||||||
|  |     user = users[0] | ||||||
|  |     print("User's ID: {}".format(user.uid)) | ||||||
|  |     print("User's name: {}".format(user.name)) | ||||||
|  |     print("User's profile picture URL: {}".format(user.photo)) | ||||||
|  |     print("User's main URL: {}".format(user.url)) | ||||||
|  |  | ||||||
|  | Since this uses Facebook's search functions, you don't have to specify the whole name, first names will usually be enough | ||||||
|  |  | ||||||
|  | You can see a full example showing all the possible ways to fetch information with ``fbchat`` by going to :ref:`examples` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. _intro_sessions: | ||||||
|  |  | ||||||
|  | Sessions | ||||||
|  | -------- | ||||||
|  |  | ||||||
|  | ``fbchat`` provides functions to retrieve and set the session cookies. | ||||||
|  | This will enable you to store the session cookies in a separate file, so that you don't have to login each time you start your script. | ||||||
|  | Use :func:`Client.getSession` to retrieve the cookies:: | ||||||
|  |  | ||||||
|  |     session_cookies = client.getSession() | ||||||
|  |  | ||||||
|  | Then you can use :func:`Client.setSession`:: | ||||||
|  |  | ||||||
|  |     client.setSession(session_cookies) | ||||||
|  |  | ||||||
|  | Or you can set the ``session_cookies`` on your initial login. | ||||||
|  | (If the session cookies are invalid, your email and password will be used to login instead):: | ||||||
|  |  | ||||||
|  |     client = Client('<email>', '<password>', session_cookies=session_cookies) | ||||||
|  |  | ||||||
|  | .. warning:: | ||||||
|  |     You session cookies can be just as valuable as you password, so store them with equal care | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. _intro_events: | ||||||
|  |  | ||||||
|  | Listening & Events | ||||||
|  | ------------------ | ||||||
|  |  | ||||||
|  | To use the listening functions ``fbchat`` offers (like :func:`Client.listen`), | ||||||
|  | you have to define what should be executed when certain events happen. | ||||||
|  | By default, (most) events will just be a `logging.info` statement, | ||||||
|  | meaning it will simply print information to the console when an event happens | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |     You can identify the event methods by their ``on`` prefix, e.g. `onMessage` | ||||||
|  |  | ||||||
|  | The event actions can be changed by subclassing the :class:`Client`, and then overwriting the event methods:: | ||||||
|  |  | ||||||
|  |     class CustomClient(Client): | ||||||
|  |         def onMessage(self, mid, author_id, message_object, thread_id, thread_type, ts, metadata, msg, **kwargs): | ||||||
|  |             # Do something with message_object here | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |     client = CustomClient('<email>', '<password>') | ||||||
|  |  | ||||||
|  | **Notice:** The following snippet is as equally valid as the previous one:: | ||||||
|  |  | ||||||
|  |     class CustomClient(Client): | ||||||
|  |         def onMessage(self, message_object, author_id, thread_id, thread_type, **kwargs): | ||||||
|  |             # Do something with message_object here | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |     client = CustomClient('<email>', '<password>') | ||||||
|  |  | ||||||
|  | The change was in the parameters that our `onMessage` method took: ``message_object`` and ``author_id`` got swapped, | ||||||
|  | and ``mid``, ``ts``, ``metadata`` and ``msg`` got removed, but the function still works, since we included ``**kwargs`` | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |     Therefore, for both backwards and forwards compatibility, | ||||||
|  |     the API actually requires that you include ``**kwargs`` as your final argument. | ||||||
|  |  | ||||||
|  | View the :ref:`examples` to see some more examples illustrating the event system | ||||||
							
								
								
									
										35
									
								
								docs/make.bat
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,35 @@ | |||||||
|  | @ECHO OFF | ||||||
|  |  | ||||||
|  | pushd %~dp0 | ||||||
|  |  | ||||||
|  | REM Command file for Sphinx documentation | ||||||
|  |  | ||||||
|  | if "%SPHINXBUILD%" == "" ( | ||||||
|  | 	set SPHINXBUILD=sphinx-build | ||||||
|  | ) | ||||||
|  | set SOURCEDIR=. | ||||||
|  | set BUILDDIR=_build | ||||||
|  |  | ||||||
|  | if "%1" == "" goto help | ||||||
|  |  | ||||||
|  | %SPHINXBUILD% >NUL 2>NUL | ||||||
|  | if errorlevel 9009 ( | ||||||
|  | 	echo. | ||||||
|  | 	echo.The 'sphinx-build' command was not found. Make sure you have Sphinx | ||||||
|  | 	echo.installed, then set the SPHINXBUILD environment variable to point | ||||||
|  | 	echo.to the full path of the 'sphinx-build' executable. Alternatively you | ||||||
|  | 	echo.may add the Sphinx directory to PATH. | ||||||
|  | 	echo. | ||||||
|  | 	echo.If you don't have Sphinx installed, grab it from | ||||||
|  | 	echo.http://sphinx-doc.org/ | ||||||
|  | 	exit /b 1 | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | %SPHINXBUILD% -M %1 %SOURCEDIR% %BUILDDIR% %SPHINXOPTS% | ||||||
|  | goto end | ||||||
|  |  | ||||||
|  | :help | ||||||
|  | %SPHINXBUILD% -M help %SOURCEDIR% %BUILDDIR% %SPHINXOPTS% | ||||||
|  |  | ||||||
|  | :end | ||||||
|  | popd | ||||||
							
								
								
									
										3
									
								
								docs/spelling/fixes.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | premade | ||||||
|  | todo | ||||||
|  | emoji | ||||||
							
								
								
									
										3
									
								
								docs/spelling/names.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | Facebook | ||||||
|  | GraphQL | ||||||
|  | GitHub | ||||||
							
								
								
									
										14
									
								
								docs/spelling/technical.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | |||||||
|  | iterables | ||||||
|  | timestamp | ||||||
|  | metadata | ||||||
|  | spam | ||||||
|  | spammy | ||||||
|  | admin | ||||||
|  | admins | ||||||
|  | unsend | ||||||
|  | unsends | ||||||
|  | unmute | ||||||
|  | spritemap | ||||||
|  | online | ||||||
|  | inbox | ||||||
|  | subclassing | ||||||
							
								
								
									
										25
									
								
								docs/testing.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | |||||||
|  | .. _testing: | ||||||
|  |  | ||||||
|  | Testing | ||||||
|  | ======= | ||||||
|  |  | ||||||
|  | To use the tests, copy ``tests/data.json`` to ``tests/my_data.json`` or type the information manually in the terminal prompts. | ||||||
|  |  | ||||||
|  | - email: Your (or a test user's) email / phone number | ||||||
|  | - password: Your (or a test user's) password | ||||||
|  | - group_thread_id: A test group that will be used to test group functionality | ||||||
|  | - user_thread_id: A person that will be used to test kick/add functionality (This user should be in the group) | ||||||
|  |  | ||||||
|  | Please remember to test all supported python versions. | ||||||
|  | If you've made any changes to the 2FA functionality, test it with a 2FA enabled account. | ||||||
|  |  | ||||||
|  | If you only want to execute specific tests, pass the function names in the command line (not including the ``test_`` prefix). Example: | ||||||
|  |  | ||||||
|  | .. code-block:: sh | ||||||
|  |  | ||||||
|  |     $ python tests.py sendMessage sessions sendEmoji | ||||||
|  |  | ||||||
|  | .. warning:: | ||||||
|  |  | ||||||
|  |     Do not execute the full set of tests in too quick succession. This can get your account temporarily blocked for spam! | ||||||
|  |     (You should execute the script at max about 10 times a day) | ||||||
							
								
								
									
										22
									
								
								docs/todo.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,22 @@ | |||||||
|  | .. _todo: | ||||||
|  |  | ||||||
|  | Todo | ||||||
|  | ==== | ||||||
|  |  | ||||||
|  | This page will be periodically updated to show missing features and documentation | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Missing Functionality | ||||||
|  | --------------------- | ||||||
|  |  | ||||||
|  | - Implement ``Client.searchForMessage`` | ||||||
|  |     - This will use the GraphQL request API | ||||||
|  | - Implement chatting with pages properly | ||||||
|  | - Write better FAQ | ||||||
|  | - Explain usage of GraphQL | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Documentation | ||||||
|  | ------------- | ||||||
|  |  | ||||||
|  | .. todolist:: | ||||||
							
								
								
									
										12
									
								
								examples/basic_usage.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  |  | ||||||
|  | from fbchat import Client | ||||||
|  | from fbchat.models import * | ||||||
|  |  | ||||||
|  | client = Client("<email>", "<password>") | ||||||
|  |  | ||||||
|  | print("Own id: {}".format(client.uid)) | ||||||
|  |  | ||||||
|  | client.send(Message(text="Hi me!"), thread_id=client.uid, thread_type=ThreadType.USER) | ||||||
|  |  | ||||||
|  | client.logout() | ||||||
							
								
								
									
										19
									
								
								examples/echobot.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  |  | ||||||
|  | from fbchat import log, Client | ||||||
|  |  | ||||||
|  | # Subclass fbchat.Client and override required methods | ||||||
|  | class EchoBot(Client): | ||||||
|  |     def onMessage(self, author_id, message_object, thread_id, thread_type, **kwargs): | ||||||
|  |         self.markAsDelivered(thread_id, message_object.uid) | ||||||
|  |         self.markAsRead(thread_id) | ||||||
|  |  | ||||||
|  |         log.info("{} from {} in {}".format(message_object, thread_id, thread_type.name)) | ||||||
|  |  | ||||||
|  |         # If you're not the author, echo | ||||||
|  |         if author_id != self.uid: | ||||||
|  |             self.send(message_object, thread_id=thread_id, thread_type=thread_type) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | client = EchoBot("<email>", "<password>") | ||||||
|  | client.listen() | ||||||
							
								
								
									
										71
									
								
								examples/fetch.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,71 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  |  | ||||||
|  | from itertools import islice | ||||||
|  | from fbchat import Client | ||||||
|  | from fbchat.models import * | ||||||
|  |  | ||||||
|  | client = Client("<email>", "<password>") | ||||||
|  |  | ||||||
|  | # Fetches a list of all users you're currently chatting with, as `User` objects | ||||||
|  | users = client.fetchAllUsers() | ||||||
|  |  | ||||||
|  | print("users' IDs: {}".format([user.uid for user in users])) | ||||||
|  | print("users' names: {}".format([user.name for user in users])) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # If we have a user id, we can use `fetchUserInfo` to fetch a `User` object | ||||||
|  | user = client.fetchUserInfo("<user id>")["<user id>"] | ||||||
|  | # We can also query both mutiple users together, which returns list of `User` objects | ||||||
|  | users = client.fetchUserInfo("<1st user id>", "<2nd user id>", "<3rd user id>") | ||||||
|  |  | ||||||
|  | print("user's name: {}".format(user.name)) | ||||||
|  | print("users' names: {}".format([users[k].name for k in users])) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # `searchForUsers` searches for the user and gives us a list of the results, | ||||||
|  | # and then we just take the first one, aka. the most likely one: | ||||||
|  | user = client.searchForUsers("<name of user>")[0] | ||||||
|  |  | ||||||
|  | print("user ID: {}".format(user.uid)) | ||||||
|  | print("user's name: {}".format(user.name)) | ||||||
|  | print("user's photo: {}".format(user.photo)) | ||||||
|  | print("Is user client's friend: {}".format(user.is_friend)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Fetches a list of the 20 top threads you're currently chatting with | ||||||
|  | threads = client.fetchThreadList() | ||||||
|  | # Fetches the next 10 threads | ||||||
|  | threads += client.fetchThreadList(offset=20, limit=10) | ||||||
|  |  | ||||||
|  | print("Threads: {}".format(threads)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Gets the last 10 messages sent to the thread | ||||||
|  | messages = client.fetchThreadMessages(thread_id="<thread id>", limit=10) | ||||||
|  | # Since the message come in reversed order, reverse them | ||||||
|  | messages.reverse() | ||||||
|  |  | ||||||
|  | # Prints the content of all the messages | ||||||
|  | for message in messages: | ||||||
|  |     print(message.text) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # If we have a thread id, we can use `fetchThreadInfo` to fetch a `Thread` object | ||||||
|  | thread = client.fetchThreadInfo("<thread id>")["<thread id>"] | ||||||
|  | print("thread's name: {}".format(thread.name)) | ||||||
|  | print("thread's type: {}".format(thread.type)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # `searchForThreads` searches works like `searchForUsers`, but gives us a list of threads instead | ||||||
|  | thread = client.searchForThreads("<name of thread>")[0] | ||||||
|  | print("thread's name: {}".format(thread.name)) | ||||||
|  | print("thread's type: {}".format(thread.type)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Here should be an example of `getUnread` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Print image url for 20 last images from thread. | ||||||
|  | images = client.fetchThreadImages("<thread id>") | ||||||
|  | for image in islice(image, 20): | ||||||
|  |     print(image.large_preview_url) | ||||||
							
								
								
									
										93
									
								
								examples/interract.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,93 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  |  | ||||||
|  | from fbchat import Client | ||||||
|  | from fbchat.models import * | ||||||
|  |  | ||||||
|  | client = Client("<email>", "<password>") | ||||||
|  |  | ||||||
|  | thread_id = "1234567890" | ||||||
|  | thread_type = ThreadType.GROUP | ||||||
|  |  | ||||||
|  | # Will send a message to the thread | ||||||
|  | client.send(Message(text="<message>"), thread_id=thread_id, thread_type=thread_type) | ||||||
|  |  | ||||||
|  | # Will send the default `like` emoji | ||||||
|  | client.send( | ||||||
|  |     Message(emoji_size=EmojiSize.LARGE), thread_id=thread_id, thread_type=thread_type | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | # Will send the emoji `👍` | ||||||
|  | client.send( | ||||||
|  |     Message(text="👍", emoji_size=EmojiSize.LARGE), | ||||||
|  |     thread_id=thread_id, | ||||||
|  |     thread_type=thread_type, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | # Will send the sticker with ID `767334476626295` | ||||||
|  | client.send( | ||||||
|  |     Message(sticker=Sticker("767334476626295")), | ||||||
|  |     thread_id=thread_id, | ||||||
|  |     thread_type=thread_type, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | # Will send a message with a mention | ||||||
|  | client.send( | ||||||
|  |     Message( | ||||||
|  |         text="This is a @mention", mentions=[Mention(thread_id, offset=10, length=8)] | ||||||
|  |     ), | ||||||
|  |     thread_id=thread_id, | ||||||
|  |     thread_type=thread_type, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | # Will send the image located at `<image path>` | ||||||
|  | client.sendLocalImage( | ||||||
|  |     "<image path>", | ||||||
|  |     message=Message(text="This is a local image"), | ||||||
|  |     thread_id=thread_id, | ||||||
|  |     thread_type=thread_type, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | # Will download the image at the URL `<image url>`, and then send it | ||||||
|  | client.sendRemoteImage( | ||||||
|  |     "<image url>", | ||||||
|  |     message=Message(text="This is a remote image"), | ||||||
|  |     thread_id=thread_id, | ||||||
|  |     thread_type=thread_type, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Only do these actions if the thread is a group | ||||||
|  | if thread_type == ThreadType.GROUP: | ||||||
|  |     # Will remove the user with ID `<user id>` from the thread | ||||||
|  |     client.removeUserFromGroup("<user id>", thread_id=thread_id) | ||||||
|  |  | ||||||
|  |     # Will add the user with ID `<user id>` to the thread | ||||||
|  |     client.addUsersToGroup("<user id>", thread_id=thread_id) | ||||||
|  |  | ||||||
|  |     # Will add the users with IDs `<1st user id>`, `<2nd user id>` and `<3th user id>` to the thread | ||||||
|  |     client.addUsersToGroup( | ||||||
|  |         ["<1st user id>", "<2nd user id>", "<3rd user id>"], thread_id=thread_id | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Will change the nickname of the user `<user_id>` to `<new nickname>` | ||||||
|  | client.changeNickname( | ||||||
|  |     "<new nickname>", "<user id>", thread_id=thread_id, thread_type=thread_type | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | # Will change the title of the thread to `<title>` | ||||||
|  | client.changeThreadTitle("<title>", thread_id=thread_id, thread_type=thread_type) | ||||||
|  |  | ||||||
|  | # Will set the typing status of the thread to `TYPING` | ||||||
|  | client.setTypingStatus( | ||||||
|  |     TypingStatus.TYPING, thread_id=thread_id, thread_type=thread_type | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | # Will change the thread color to `MESSENGER_BLUE` | ||||||
|  | client.changeThreadColor(ThreadColor.MESSENGER_BLUE, thread_id=thread_id) | ||||||
|  |  | ||||||
|  | # Will change the thread emoji to `👍` | ||||||
|  | client.changeThreadEmoji("👍", thread_id=thread_id) | ||||||
|  |  | ||||||
|  | # Will react to a message with a 😍 emoji | ||||||
|  | client.reactToMessage("<message id>", MessageReaction.LOVE) | ||||||
							
								
								
									
										83
									
								
								examples/keepbot.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,83 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  |  | ||||||
|  | from fbchat import log, Client | ||||||
|  | from fbchat.models import * | ||||||
|  |  | ||||||
|  | # Change this to your group id | ||||||
|  | old_thread_id = "1234567890" | ||||||
|  |  | ||||||
|  | # Change these to match your liking | ||||||
|  | old_color = ThreadColor.MESSENGER_BLUE | ||||||
|  | old_emoji = "👍" | ||||||
|  | old_title = "Old group chat name" | ||||||
|  | old_nicknames = { | ||||||
|  |     "12345678901": "User nr. 1's nickname", | ||||||
|  |     "12345678902": "User nr. 2's nickname", | ||||||
|  |     "12345678903": "User nr. 3's nickname", | ||||||
|  |     "12345678904": "User nr. 4's nickname", | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class KeepBot(Client): | ||||||
|  |     def onColorChange(self, author_id, new_color, thread_id, thread_type, **kwargs): | ||||||
|  |         if old_thread_id == thread_id and old_color != new_color: | ||||||
|  |             log.info( | ||||||
|  |                 "{} changed the thread color. It will be changed back".format(author_id) | ||||||
|  |             ) | ||||||
|  |             self.changeThreadColor(old_color, thread_id=thread_id) | ||||||
|  |  | ||||||
|  |     def onEmojiChange(self, author_id, new_emoji, thread_id, thread_type, **kwargs): | ||||||
|  |         if old_thread_id == thread_id and new_emoji != old_emoji: | ||||||
|  |             log.info( | ||||||
|  |                 "{} changed the thread emoji. It will be changed back".format(author_id) | ||||||
|  |             ) | ||||||
|  |             self.changeThreadEmoji(old_emoji, thread_id=thread_id) | ||||||
|  |  | ||||||
|  |     def onPeopleAdded(self, added_ids, author_id, thread_id, **kwargs): | ||||||
|  |         if old_thread_id == thread_id and author_id != self.uid: | ||||||
|  |             log.info("{} got added. They will be removed".format(added_ids)) | ||||||
|  |             for added_id in added_ids: | ||||||
|  |                 self.removeUserFromGroup(added_id, thread_id=thread_id) | ||||||
|  |  | ||||||
|  |     def onPersonRemoved(self, removed_id, author_id, thread_id, **kwargs): | ||||||
|  |         # No point in trying to add ourself | ||||||
|  |         if ( | ||||||
|  |             old_thread_id == thread_id | ||||||
|  |             and removed_id != self.uid | ||||||
|  |             and author_id != self.uid | ||||||
|  |         ): | ||||||
|  |             log.info("{} got removed. They will be re-added".format(removed_id)) | ||||||
|  |             self.addUsersToGroup(removed_id, thread_id=thread_id) | ||||||
|  |  | ||||||
|  |     def onTitleChange(self, author_id, new_title, thread_id, thread_type, **kwargs): | ||||||
|  |         if old_thread_id == thread_id and old_title != new_title: | ||||||
|  |             log.info( | ||||||
|  |                 "{} changed the thread title. It will be changed back".format(author_id) | ||||||
|  |             ) | ||||||
|  |             self.changeThreadTitle( | ||||||
|  |                 old_title, thread_id=thread_id, thread_type=thread_type | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |     def onNicknameChange( | ||||||
|  |         self, author_id, changed_for, new_nickname, thread_id, thread_type, **kwargs | ||||||
|  |     ): | ||||||
|  |         if ( | ||||||
|  |             old_thread_id == thread_id | ||||||
|  |             and changed_for in old_nicknames | ||||||
|  |             and old_nicknames[changed_for] != new_nickname | ||||||
|  |         ): | ||||||
|  |             log.info( | ||||||
|  |                 "{} changed {}'s' nickname. It will be changed back".format( | ||||||
|  |                     author_id, changed_for | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             self.changeNickname( | ||||||
|  |                 old_nicknames[changed_for], | ||||||
|  |                 changed_for, | ||||||
|  |                 thread_id=thread_id, | ||||||
|  |                 thread_type=thread_type, | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | client = KeepBot("<email>", "<password>") | ||||||
|  | client.listen() | ||||||
							
								
								
									
										25
									
								
								examples/removebot.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  |  | ||||||
|  | from fbchat import log, Client | ||||||
|  | from fbchat.models import * | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RemoveBot(Client): | ||||||
|  |     def onMessage(self, author_id, message_object, thread_id, thread_type, **kwargs): | ||||||
|  |         # We can only kick people from group chats, so no need to try if it's a user chat | ||||||
|  |         if message_object.text == "Remove me!" and thread_type == ThreadType.GROUP: | ||||||
|  |             log.info("{} will be removed from {}".format(author_id, thread_id)) | ||||||
|  |             self.removeUserFromGroup(author_id, thread_id=thread_id) | ||||||
|  |         else: | ||||||
|  |             # Sends the data to the inherited onMessage, so that we can still see when a message is recieved | ||||||
|  |             super(RemoveBot, self).onMessage( | ||||||
|  |                 author_id=author_id, | ||||||
|  |                 message_object=message_object, | ||||||
|  |                 thread_id=thread_id, | ||||||
|  |                 thread_type=thread_type, | ||||||
|  |                 **kwargs | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | client = RemoveBot("<email>", "<password>") | ||||||
|  | client.listen() | ||||||
| @@ -1,26 +1,25 @@ | |||||||
| # -*- coding: UTF-8 -*- | # -*- coding: UTF-8 -*- | ||||||
|  | """Facebook Chat (Messenger) for Python | ||||||
|  |  | ||||||
|  | :copyright: (c) 2015 - 2019 by Taehoon Kim | ||||||
|  | :license: BSD 3-Clause, see LICENSE for more details. | ||||||
| """ | """ | ||||||
|     fbchat | from __future__ import unicode_literals | ||||||
|     ~~~~~~ |  | ||||||
|  |  | ||||||
|     Facebook Chat (Messenger) for Python | # These imports are far too general, but they're needed for backwards compatbility. | ||||||
|  | from .models import * | ||||||
|  |  | ||||||
|     :copyright: (c) 2015 by Taehoon Kim. | from ._client import Client | ||||||
|     :license: BSD, see LICENSE for more details. | from ._util import log  # TODO: Remove this (from examples too) | ||||||
| """ |  | ||||||
|  |  | ||||||
|  | __title__ = "fbchat" | ||||||
|  | __version__ = "1.9.6" | ||||||
|  | __description__ = "Facebook Chat (Messenger) for Python" | ||||||
|  |  | ||||||
| from .client import * | __copyright__ = "Copyright 2015 - 2019 by Taehoon Kim" | ||||||
|  | __license__ = "BSD 3-Clause" | ||||||
|  |  | ||||||
|  | __author__ = "Taehoon Kim; Moreels Pieter-Jan; Mads Marquart" | ||||||
|  | __email__ = "carpedm20@gmail.com" | ||||||
|  |  | ||||||
| __copyright__ = 'Copyright 2015 by Taehoon Kim' | __all__ = ["Client"] | ||||||
| __version__ = '0.9.3' |  | ||||||
| __license__ = 'BSD' |  | ||||||
| __author__ = 'Taehoon Kim; Moreels Pieter-Jan' |  | ||||||
| __email__ = 'carpedm20@gmail.com' |  | ||||||
| __source__ = 'https://github.com/carpedm20/fbchat/' |  | ||||||
|  |  | ||||||
| __all__ = [ |  | ||||||
|     'Client', |  | ||||||
| ] |  | ||||||
|   | |||||||
							
								
								
									
										86
									
								
								fbchat/_attachment.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,86 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | from . import _util | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class Attachment(object): | ||||||
|  |     """Represents a Facebook attachment.""" | ||||||
|  |  | ||||||
|  |     #: The attachment ID | ||||||
|  |     uid = attr.ib(None) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class UnsentMessage(Attachment): | ||||||
|  |     """Represents an unsent message attachment.""" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class ShareAttachment(Attachment): | ||||||
|  |     """Represents a shared item (e.g. URL) attachment.""" | ||||||
|  |  | ||||||
|  |     #: ID of the author of the shared post | ||||||
|  |     author = attr.ib(None) | ||||||
|  |     #: Target URL | ||||||
|  |     url = attr.ib(None) | ||||||
|  |     #: Original URL if Facebook redirects the URL | ||||||
|  |     original_url = attr.ib(None) | ||||||
|  |     #: Title of the attachment | ||||||
|  |     title = attr.ib(None) | ||||||
|  |     #: Description of the attachment | ||||||
|  |     description = attr.ib(None) | ||||||
|  |     #: Name of the source | ||||||
|  |     source = attr.ib(None) | ||||||
|  |     #: URL of the attachment image | ||||||
|  |     image_url = attr.ib(None) | ||||||
|  |     #: URL of the original image if Facebook uses ``safe_image`` | ||||||
|  |     original_image_url = attr.ib(None) | ||||||
|  |     #: Width of the image | ||||||
|  |     image_width = attr.ib(None) | ||||||
|  |     #: Height of the image | ||||||
|  |     image_height = attr.ib(None) | ||||||
|  |     #: List of additional attachments | ||||||
|  |     attachments = attr.ib(factory=list, converter=lambda x: [] if x is None else x) | ||||||
|  |  | ||||||
|  |     # Put here for backwards compatibility, so that the init argument order is preserved | ||||||
|  |     uid = attr.ib(None) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         from . import _file | ||||||
|  |  | ||||||
|  |         url = data.get("url") | ||||||
|  |         rtn = cls( | ||||||
|  |             uid=data.get("deduplication_key"), | ||||||
|  |             author=data["target"]["actors"][0]["id"] | ||||||
|  |             if data["target"].get("actors") | ||||||
|  |             else None, | ||||||
|  |             url=url, | ||||||
|  |             original_url=_util.get_url_parameter(url, "u") | ||||||
|  |             if "/l.php?u=" in url | ||||||
|  |             else url, | ||||||
|  |             title=data["title_with_entities"].get("text"), | ||||||
|  |             description=data["description"].get("text") | ||||||
|  |             if data.get("description") | ||||||
|  |             else None, | ||||||
|  |             source=data["source"].get("text") if data.get("source") else None, | ||||||
|  |             attachments=[ | ||||||
|  |                 _file.graphql_to_subattachment(attachment) | ||||||
|  |                 for attachment in data.get("subattachments") | ||||||
|  |             ], | ||||||
|  |         ) | ||||||
|  |         media = data.get("media") | ||||||
|  |         if media and media.get("image"): | ||||||
|  |             image = media["image"] | ||||||
|  |             rtn.image_url = image.get("uri") | ||||||
|  |             rtn.original_image_url = ( | ||||||
|  |                 _util.get_url_parameter(rtn.image_url, "url") | ||||||
|  |                 if "/safe_image.php" in rtn.image_url | ||||||
|  |                 else rtn.image_url | ||||||
|  |             ) | ||||||
|  |             rtn.image_width = image.get("width") | ||||||
|  |             rtn.image_height = image.get("height") | ||||||
|  |         return rtn | ||||||
							
								
								
									
										3875
									
								
								fbchat/_client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										26
									
								
								fbchat/_core.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import logging | ||||||
|  | import aenum | ||||||
|  |  | ||||||
|  | log = logging.getLogger("client") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Enum(aenum.Enum): | ||||||
|  |     """Used internally by ``fbchat`` to support enumerations""" | ||||||
|  |  | ||||||
|  |     def __repr__(self): | ||||||
|  |         # For documentation: | ||||||
|  |         return "{}.{}".format(type(self).__name__, self.name) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _extend_if_invalid(cls, value): | ||||||
|  |         try: | ||||||
|  |             return cls(value) | ||||||
|  |         except ValueError: | ||||||
|  |             log.warning( | ||||||
|  |                 "Failed parsing {.__name__}({!r}). Extending enum.".format(cls, value) | ||||||
|  |             ) | ||||||
|  |             aenum.extend_enum(cls, "UNKNOWN_{}".format(value).upper(), value) | ||||||
|  |             return cls(value) | ||||||
							
								
								
									
										60
									
								
								fbchat/_exception.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,60 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FBchatException(Exception): | ||||||
|  |     """Custom exception thrown by ``fbchat``. | ||||||
|  |  | ||||||
|  |     All exceptions in the ``fbchat`` module inherits this. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FBchatFacebookError(FBchatException): | ||||||
|  |     #: The error code that Facebook returned | ||||||
|  |     fb_error_code = None | ||||||
|  |     #: The error message that Facebook returned (In the user's own language) | ||||||
|  |     fb_error_message = None | ||||||
|  |     #: The status code that was sent in the HTTP response (e.g. 404) (Usually only set if not successful, aka. not 200) | ||||||
|  |     request_status_code = None | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         message, | ||||||
|  |         fb_error_code=None, | ||||||
|  |         fb_error_message=None, | ||||||
|  |         request_status_code=None, | ||||||
|  |     ): | ||||||
|  |         super(FBchatFacebookError, self).__init__(message) | ||||||
|  |         """Thrown by ``fbchat`` when Facebook returns an error""" | ||||||
|  |         self.fb_error_code = str(fb_error_code) | ||||||
|  |         self.fb_error_message = fb_error_message | ||||||
|  |         self.request_status_code = request_status_code | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FBchatInvalidParameters(FBchatFacebookError): | ||||||
|  |     """Raised by Facebook if: | ||||||
|  |  | ||||||
|  |     - Some function supplied invalid parameters. | ||||||
|  |     - Some content is not found. | ||||||
|  |     - Some content is no longer available. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FBchatNotLoggedIn(FBchatFacebookError): | ||||||
|  |     """Raised by Facebook if the client has been logged out.""" | ||||||
|  |  | ||||||
|  |     fb_error_code = "1357001" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FBchatPleaseRefresh(FBchatFacebookError): | ||||||
|  |     """Raised by Facebook if the client has been inactive for too long. | ||||||
|  |  | ||||||
|  |     This error usually happens after 1-2 days of inactivity. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     fb_error_code = "1357004" | ||||||
|  |     fb_error_message = "Please try closing and re-opening your browser window." | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FBchatUserError(FBchatException): | ||||||
|  |     """Thrown by ``fbchat`` when wrong values are entered.""" | ||||||
							
								
								
									
										301
									
								
								fbchat/_file.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,301 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | from ._attachment import Attachment | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class FileAttachment(Attachment): | ||||||
|  |     """Represents a file that has been sent as a Facebook attachment.""" | ||||||
|  |  | ||||||
|  |     #: URL where you can download the file | ||||||
|  |     url = attr.ib(None) | ||||||
|  |     #: Size of the file in bytes | ||||||
|  |     size = attr.ib(None) | ||||||
|  |     #: Name of the file | ||||||
|  |     name = attr.ib(None) | ||||||
|  |     #: Whether Facebook determines that this file may be harmful | ||||||
|  |     is_malicious = attr.ib(None) | ||||||
|  |  | ||||||
|  |     # Put here for backwards compatibility, so that the init argument order is preserved | ||||||
|  |     uid = attr.ib(None) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         return cls( | ||||||
|  |             url=data.get("url"), | ||||||
|  |             name=data.get("filename"), | ||||||
|  |             is_malicious=data.get("is_malicious"), | ||||||
|  |             uid=data.get("message_file_fbid"), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class AudioAttachment(Attachment): | ||||||
|  |     """Represents an audio file that has been sent as a Facebook attachment.""" | ||||||
|  |  | ||||||
|  |     #: Name of the file | ||||||
|  |     filename = attr.ib(None) | ||||||
|  |     #: URL of the audio file | ||||||
|  |     url = attr.ib(None) | ||||||
|  |     #: Duration of the audio clip in milliseconds | ||||||
|  |     duration = attr.ib(None) | ||||||
|  |     #: Audio type | ||||||
|  |     audio_type = attr.ib(None) | ||||||
|  |  | ||||||
|  |     # Put here for backwards compatibility, so that the init argument order is preserved | ||||||
|  |     uid = attr.ib(None) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         return cls( | ||||||
|  |             filename=data.get("filename"), | ||||||
|  |             url=data.get("playable_url"), | ||||||
|  |             duration=data.get("playable_duration_in_ms"), | ||||||
|  |             audio_type=data.get("audio_type"), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class ImageAttachment(Attachment): | ||||||
|  |     """Represents an image that has been sent as a Facebook attachment. | ||||||
|  |  | ||||||
|  |     To retrieve the full image URL, use: `Client.fetchImageUrl`, and pass it the id of | ||||||
|  |     the image attachment. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     #: The extension of the original image (e.g. ``png``) | ||||||
|  |     original_extension = attr.ib(None) | ||||||
|  |     #: Width of original image | ||||||
|  |     width = attr.ib(None, converter=lambda x: None if x is None else int(x)) | ||||||
|  |     #: Height of original image | ||||||
|  |     height = attr.ib(None, converter=lambda x: None if x is None else int(x)) | ||||||
|  |  | ||||||
|  |     #: Whether the image is animated | ||||||
|  |     is_animated = attr.ib(None) | ||||||
|  |  | ||||||
|  |     #: URL to a thumbnail of the image | ||||||
|  |     thumbnail_url = attr.ib(None) | ||||||
|  |  | ||||||
|  |     #: URL to a medium preview of the image | ||||||
|  |     preview_url = attr.ib(None) | ||||||
|  |     #: Width of the medium preview image | ||||||
|  |     preview_width = attr.ib(None) | ||||||
|  |     #: Height of the medium preview image | ||||||
|  |     preview_height = attr.ib(None) | ||||||
|  |  | ||||||
|  |     #: URL to a large preview of the image | ||||||
|  |     large_preview_url = attr.ib(None) | ||||||
|  |     #: Width of the large preview image | ||||||
|  |     large_preview_width = attr.ib(None) | ||||||
|  |     #: Height of the large preview image | ||||||
|  |     large_preview_height = attr.ib(None) | ||||||
|  |  | ||||||
|  |     #: URL to an animated preview of the image (e.g. for GIFs) | ||||||
|  |     animated_preview_url = attr.ib(None) | ||||||
|  |     #: Width of the animated preview image | ||||||
|  |     animated_preview_width = attr.ib(None) | ||||||
|  |     #: Height of the animated preview image | ||||||
|  |     animated_preview_height = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         original_extension=None, | ||||||
|  |         width=None, | ||||||
|  |         height=None, | ||||||
|  |         is_animated=None, | ||||||
|  |         thumbnail_url=None, | ||||||
|  |         preview=None, | ||||||
|  |         large_preview=None, | ||||||
|  |         animated_preview=None, | ||||||
|  |         **kwargs | ||||||
|  |     ): | ||||||
|  |         super(ImageAttachment, self).__init__(**kwargs) | ||||||
|  |         self.original_extension = original_extension | ||||||
|  |         if width is not None: | ||||||
|  |             width = int(width) | ||||||
|  |         self.width = width | ||||||
|  |         if height is not None: | ||||||
|  |             height = int(height) | ||||||
|  |         self.height = height | ||||||
|  |         self.is_animated = is_animated | ||||||
|  |         self.thumbnail_url = thumbnail_url | ||||||
|  |  | ||||||
|  |         if preview is None: | ||||||
|  |             preview = {} | ||||||
|  |         self.preview_url = preview.get("uri") | ||||||
|  |         self.preview_width = preview.get("width") | ||||||
|  |         self.preview_height = preview.get("height") | ||||||
|  |  | ||||||
|  |         if large_preview is None: | ||||||
|  |             large_preview = {} | ||||||
|  |         self.large_preview_url = large_preview.get("uri") | ||||||
|  |         self.large_preview_width = large_preview.get("width") | ||||||
|  |         self.large_preview_height = large_preview.get("height") | ||||||
|  |  | ||||||
|  |         if animated_preview is None: | ||||||
|  |             animated_preview = {} | ||||||
|  |         self.animated_preview_url = animated_preview.get("uri") | ||||||
|  |         self.animated_preview_width = animated_preview.get("width") | ||||||
|  |         self.animated_preview_height = animated_preview.get("height") | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         return cls( | ||||||
|  |             original_extension=data.get("original_extension") | ||||||
|  |             or (data["filename"].split("-")[0] if data.get("filename") else None), | ||||||
|  |             width=data.get("original_dimensions", {}).get("width"), | ||||||
|  |             height=data.get("original_dimensions", {}).get("height"), | ||||||
|  |             is_animated=data["__typename"] == "MessageAnimatedImage", | ||||||
|  |             thumbnail_url=data.get("thumbnail", {}).get("uri"), | ||||||
|  |             preview=data.get("preview") or data.get("preview_image"), | ||||||
|  |             large_preview=data.get("large_preview"), | ||||||
|  |             animated_preview=data.get("animated_image"), | ||||||
|  |             uid=data.get("legacy_attachment_id"), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_list(cls, data): | ||||||
|  |         data = data["node"] | ||||||
|  |         return cls( | ||||||
|  |             width=data["original_dimensions"].get("x"), | ||||||
|  |             height=data["original_dimensions"].get("y"), | ||||||
|  |             thumbnail_url=data["image"].get("uri"), | ||||||
|  |             large_preview=data["image2"], | ||||||
|  |             preview=data["image1"], | ||||||
|  |             uid=data["legacy_attachment_id"], | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class VideoAttachment(Attachment): | ||||||
|  |     """Represents a video that has been sent as a Facebook attachment.""" | ||||||
|  |  | ||||||
|  |     #: Size of the original video in bytes | ||||||
|  |     size = attr.ib(None) | ||||||
|  |     #: Width of original video | ||||||
|  |     width = attr.ib(None) | ||||||
|  |     #: Height of original video | ||||||
|  |     height = attr.ib(None) | ||||||
|  |     #: Length of video in milliseconds | ||||||
|  |     duration = attr.ib(None) | ||||||
|  |     #: URL to very compressed preview video | ||||||
|  |     preview_url = attr.ib(None) | ||||||
|  |  | ||||||
|  |     #: URL to a small preview image of the video | ||||||
|  |     small_image_url = attr.ib(None) | ||||||
|  |     #: Width of the small preview image | ||||||
|  |     small_image_width = attr.ib(None) | ||||||
|  |     #: Height of the small preview image | ||||||
|  |     small_image_height = attr.ib(None) | ||||||
|  |  | ||||||
|  |     #: URL to a medium preview image of the video | ||||||
|  |     medium_image_url = attr.ib(None) | ||||||
|  |     #: Width of the medium preview image | ||||||
|  |     medium_image_width = attr.ib(None) | ||||||
|  |     #: Height of the medium preview image | ||||||
|  |     medium_image_height = attr.ib(None) | ||||||
|  |  | ||||||
|  |     #: URL to a large preview image of the video | ||||||
|  |     large_image_url = attr.ib(None) | ||||||
|  |     #: Width of the large preview image | ||||||
|  |     large_image_width = attr.ib(None) | ||||||
|  |     #: Height of the large preview image | ||||||
|  |     large_image_height = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         size=None, | ||||||
|  |         width=None, | ||||||
|  |         height=None, | ||||||
|  |         duration=None, | ||||||
|  |         preview_url=None, | ||||||
|  |         small_image=None, | ||||||
|  |         medium_image=None, | ||||||
|  |         large_image=None, | ||||||
|  |         **kwargs | ||||||
|  |     ): | ||||||
|  |         super(VideoAttachment, self).__init__(**kwargs) | ||||||
|  |         self.size = size | ||||||
|  |         self.width = width | ||||||
|  |         self.height = height | ||||||
|  |         self.duration = duration | ||||||
|  |         self.preview_url = preview_url | ||||||
|  |  | ||||||
|  |         if small_image is None: | ||||||
|  |             small_image = {} | ||||||
|  |         self.small_image_url = small_image.get("uri") | ||||||
|  |         self.small_image_width = small_image.get("width") | ||||||
|  |         self.small_image_height = small_image.get("height") | ||||||
|  |  | ||||||
|  |         if medium_image is None: | ||||||
|  |             medium_image = {} | ||||||
|  |         self.medium_image_url = medium_image.get("uri") | ||||||
|  |         self.medium_image_width = medium_image.get("width") | ||||||
|  |         self.medium_image_height = medium_image.get("height") | ||||||
|  |  | ||||||
|  |         if large_image is None: | ||||||
|  |             large_image = {} | ||||||
|  |         self.large_image_url = large_image.get("uri") | ||||||
|  |         self.large_image_width = large_image.get("width") | ||||||
|  |         self.large_image_height = large_image.get("height") | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         return cls( | ||||||
|  |             width=data.get("original_dimensions", {}).get("width"), | ||||||
|  |             height=data.get("original_dimensions", {}).get("height"), | ||||||
|  |             duration=data.get("playable_duration_in_ms"), | ||||||
|  |             preview_url=data.get("playable_url"), | ||||||
|  |             small_image=data.get("chat_image"), | ||||||
|  |             medium_image=data.get("inbox_image"), | ||||||
|  |             large_image=data.get("large_image"), | ||||||
|  |             uid=data.get("legacy_attachment_id"), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_subattachment(cls, data): | ||||||
|  |         media = data["media"] | ||||||
|  |         return cls( | ||||||
|  |             duration=media.get("playable_duration_in_ms"), | ||||||
|  |             preview_url=media.get("playable_url"), | ||||||
|  |             medium_image=media.get("image"), | ||||||
|  |             uid=data["target"].get("video_id"), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_list(cls, data): | ||||||
|  |         data = data["node"] | ||||||
|  |         return cls( | ||||||
|  |             width=data["original_dimensions"].get("x"), | ||||||
|  |             height=data["original_dimensions"].get("y"), | ||||||
|  |             small_image=data["image"], | ||||||
|  |             medium_image=data["image1"], | ||||||
|  |             large_image=data["image2"], | ||||||
|  |             uid=data["legacy_attachment_id"], | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def graphql_to_attachment(data): | ||||||
|  |     _type = data["__typename"] | ||||||
|  |     if _type in ["MessageImage", "MessageAnimatedImage"]: | ||||||
|  |         return ImageAttachment._from_graphql(data) | ||||||
|  |     elif _type == "MessageVideo": | ||||||
|  |         return VideoAttachment._from_graphql(data) | ||||||
|  |     elif _type == "MessageAudio": | ||||||
|  |         return AudioAttachment._from_graphql(data) | ||||||
|  |     elif _type == "MessageFile": | ||||||
|  |         return FileAttachment._from_graphql(data) | ||||||
|  |  | ||||||
|  |     return Attachment(uid=data.get("legacy_attachment_id")) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def graphql_to_subattachment(data): | ||||||
|  |     target = data.get("target") | ||||||
|  |     type_ = target.get("__typename") if target else None | ||||||
|  |  | ||||||
|  |     if type_ == "Video": | ||||||
|  |         return VideoAttachment._from_subattachment(data) | ||||||
|  |  | ||||||
|  |     return None | ||||||
							
								
								
									
										237
									
								
								fbchat/_graphql.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,237 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import json | ||||||
|  | import re | ||||||
|  | from . import _util | ||||||
|  | from ._exception import FBchatException | ||||||
|  |  | ||||||
|  | # Shameless copy from https://stackoverflow.com/a/8730674 | ||||||
|  | FLAGS = re.VERBOSE | re.MULTILINE | re.DOTALL | ||||||
|  | WHITESPACE = re.compile(r"[ \t\n\r]*", FLAGS) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ConcatJSONDecoder(json.JSONDecoder): | ||||||
|  |     def decode(self, s, _w=WHITESPACE.match): | ||||||
|  |         s_len = len(s) | ||||||
|  |  | ||||||
|  |         objs = [] | ||||||
|  |         end = 0 | ||||||
|  |         while end != s_len: | ||||||
|  |             obj, end = self.raw_decode(s, idx=_w(s, end).end()) | ||||||
|  |             end = _w(s, end).end() | ||||||
|  |             objs.append(obj) | ||||||
|  |         return objs | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # End shameless copy | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def queries_to_json(*queries): | ||||||
|  |     """ | ||||||
|  |     Queries should be a list of GraphQL objects | ||||||
|  |     """ | ||||||
|  |     rtn = {} | ||||||
|  |     for i, query in enumerate(queries): | ||||||
|  |         rtn["q{}".format(i)] = query | ||||||
|  |     return json.dumps(rtn) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def response_to_json(content): | ||||||
|  |     content = _util.strip_json_cruft(content)  # Usually only needed in some error cases | ||||||
|  |     try: | ||||||
|  |         j = json.loads(content, cls=ConcatJSONDecoder) | ||||||
|  |     except Exception: | ||||||
|  |         raise FBchatException("Error while parsing JSON: {}".format(repr(content))) | ||||||
|  |  | ||||||
|  |     rtn = [None] * (len(j)) | ||||||
|  |     for x in j: | ||||||
|  |         if "error_results" in x: | ||||||
|  |             del rtn[-1] | ||||||
|  |             continue | ||||||
|  |         _util.handle_payload_error(x) | ||||||
|  |         [(key, value)] = x.items() | ||||||
|  |         _util.handle_graphql_errors(value) | ||||||
|  |         if "response" in value: | ||||||
|  |             rtn[int(key[1:])] = value["response"] | ||||||
|  |         else: | ||||||
|  |             rtn[int(key[1:])] = value["data"] | ||||||
|  |  | ||||||
|  |     _util.log.debug(rtn) | ||||||
|  |  | ||||||
|  |     return rtn | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def from_query(query, params): | ||||||
|  |     return {"priority": 0, "q": query, "query_params": params} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def from_query_id(query_id, params): | ||||||
|  |     return {"query_id": query_id, "query_params": params} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def from_doc(doc, params): | ||||||
|  |     return {"doc": doc, "query_params": params} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def from_doc_id(doc_id, params): | ||||||
|  |     return {"doc_id": doc_id, "query_params": params} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | FRAGMENT_USER = """ | ||||||
|  | QueryFragment User: User { | ||||||
|  |     id, | ||||||
|  |     name, | ||||||
|  |     first_name, | ||||||
|  |     last_name, | ||||||
|  |     profile_picture.width(<pic_size>).height(<pic_size>) { | ||||||
|  |         uri | ||||||
|  |     }, | ||||||
|  |     is_viewer_friend, | ||||||
|  |     url, | ||||||
|  |     gender, | ||||||
|  |     viewer_affinity | ||||||
|  | } | ||||||
|  | """ | ||||||
|  |  | ||||||
|  | FRAGMENT_GROUP = """ | ||||||
|  | QueryFragment Group: MessageThread { | ||||||
|  |     name, | ||||||
|  |     thread_key { | ||||||
|  |         thread_fbid | ||||||
|  |     }, | ||||||
|  |     image { | ||||||
|  |         uri | ||||||
|  |     }, | ||||||
|  |     is_group_thread, | ||||||
|  |     all_participants { | ||||||
|  |         nodes { | ||||||
|  |             messaging_actor { | ||||||
|  |                 id | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     customization_info { | ||||||
|  |         participant_customizations { | ||||||
|  |             participant_id, | ||||||
|  |             nickname | ||||||
|  |         }, | ||||||
|  |         outgoing_bubble_color, | ||||||
|  |         emoji | ||||||
|  |     }, | ||||||
|  |     thread_admins { | ||||||
|  |         id | ||||||
|  |     }, | ||||||
|  |     group_approval_queue { | ||||||
|  |         nodes { | ||||||
|  |             requester { | ||||||
|  |                 id | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     approval_mode, | ||||||
|  |     joinable_mode { | ||||||
|  |         mode, | ||||||
|  |         link | ||||||
|  |     }, | ||||||
|  |     event_reminders { | ||||||
|  |         nodes { | ||||||
|  |             id, | ||||||
|  |             lightweight_event_creator { | ||||||
|  |                 id | ||||||
|  |             }, | ||||||
|  |             time, | ||||||
|  |             location_name, | ||||||
|  |             event_title, | ||||||
|  |             event_reminder_members { | ||||||
|  |                 edges { | ||||||
|  |                     node { | ||||||
|  |                         id | ||||||
|  |                     }, | ||||||
|  |                     guest_list_state | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | """ | ||||||
|  |  | ||||||
|  | FRAGMENT_PAGE = """ | ||||||
|  | QueryFragment Page: Page { | ||||||
|  |     id, | ||||||
|  |     name, | ||||||
|  |     profile_picture.width(32).height(32) { | ||||||
|  |         uri | ||||||
|  |     }, | ||||||
|  |     url, | ||||||
|  |     category_type, | ||||||
|  |     city { | ||||||
|  |         name | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | """ | ||||||
|  |  | ||||||
|  | SEARCH_USER = ( | ||||||
|  |     """ | ||||||
|  | Query SearchUser(<search> = '', <limit> = 10) { | ||||||
|  |     entities_named(<search>) { | ||||||
|  |         search_results.of_type(user).first(<limit>) as users { | ||||||
|  |             nodes { | ||||||
|  |                 @User | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | """ | ||||||
|  |     + FRAGMENT_USER | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | SEARCH_GROUP = ( | ||||||
|  |     """ | ||||||
|  | Query SearchGroup(<search> = '', <limit> = 10, <pic_size> = 32) { | ||||||
|  |     viewer() { | ||||||
|  |         message_threads.with_thread_name(<search>).last(<limit>) as groups { | ||||||
|  |             nodes { | ||||||
|  |                 @Group | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | """ | ||||||
|  |     + FRAGMENT_GROUP | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | SEARCH_PAGE = ( | ||||||
|  |     """ | ||||||
|  | Query SearchPage(<search> = '', <limit> = 10) { | ||||||
|  |     entities_named(<search>) { | ||||||
|  |         search_results.of_type(page).first(<limit>) as pages { | ||||||
|  |             nodes { | ||||||
|  |                 @Page | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | """ | ||||||
|  |     + FRAGMENT_PAGE | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | SEARCH_THREAD = ( | ||||||
|  |     """ | ||||||
|  | Query SearchThread(<search> = '', <limit> = 10) { | ||||||
|  |     entities_named(<search>) { | ||||||
|  |         search_results.first(<limit>) as threads { | ||||||
|  |             nodes { | ||||||
|  |                 __typename, | ||||||
|  |                 @User, | ||||||
|  |                 @Group, | ||||||
|  |                 @Page | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | """ | ||||||
|  |     + FRAGMENT_USER | ||||||
|  |     + FRAGMENT_GROUP | ||||||
|  |     + FRAGMENT_PAGE | ||||||
|  | ) | ||||||
							
								
								
									
										121
									
								
								fbchat/_group.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,121 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | from . import _plan | ||||||
|  | from ._thread import ThreadType, Thread | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class Group(Thread): | ||||||
|  |     """Represents a Facebook group. Inherits `Thread`.""" | ||||||
|  |  | ||||||
|  |     #: Unique list (set) of the group thread's participant user IDs | ||||||
|  |     participants = attr.ib(factory=set, converter=lambda x: set() if x is None else x) | ||||||
|  |     #: A dictionary, containing user nicknames mapped to their IDs | ||||||
|  |     nicknames = attr.ib(factory=dict, converter=lambda x: {} if x is None else x) | ||||||
|  |     #: A :class:`ThreadColor`. The groups's message color | ||||||
|  |     color = attr.ib(None) | ||||||
|  |     #: The groups's default emoji | ||||||
|  |     emoji = attr.ib(None) | ||||||
|  |     # Set containing user IDs of thread admins | ||||||
|  |     admins = attr.ib(factory=set, converter=lambda x: set() if x is None else x) | ||||||
|  |     # True if users need approval to join | ||||||
|  |     approval_mode = attr.ib(None) | ||||||
|  |     # Set containing user IDs requesting to join | ||||||
|  |     approval_requests = attr.ib( | ||||||
|  |         factory=set, converter=lambda x: set() if x is None else x | ||||||
|  |     ) | ||||||
|  |     # Link for joining group | ||||||
|  |     join_link = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         uid, | ||||||
|  |         participants=None, | ||||||
|  |         nicknames=None, | ||||||
|  |         color=None, | ||||||
|  |         emoji=None, | ||||||
|  |         admins=None, | ||||||
|  |         approval_mode=None, | ||||||
|  |         approval_requests=None, | ||||||
|  |         join_link=None, | ||||||
|  |         privacy_mode=None, | ||||||
|  |         **kwargs | ||||||
|  |     ): | ||||||
|  |         super(Group, self).__init__(ThreadType.GROUP, uid, **kwargs) | ||||||
|  |         if participants is None: | ||||||
|  |             participants = set() | ||||||
|  |         self.participants = participants | ||||||
|  |         if nicknames is None: | ||||||
|  |             nicknames = [] | ||||||
|  |         self.nicknames = nicknames | ||||||
|  |         self.color = color | ||||||
|  |         self.emoji = emoji | ||||||
|  |         if admins is None: | ||||||
|  |             admins = set() | ||||||
|  |         self.admins = admins | ||||||
|  |         self.approval_mode = approval_mode | ||||||
|  |         if approval_requests is None: | ||||||
|  |             approval_requests = set() | ||||||
|  |         self.approval_requests = approval_requests | ||||||
|  |         self.join_link = join_link | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         if data.get("image") is None: | ||||||
|  |             data["image"] = {} | ||||||
|  |         c_info = cls._parse_customization_info(data) | ||||||
|  |         last_message_timestamp = None | ||||||
|  |         if "last_message" in data: | ||||||
|  |             last_message_timestamp = data["last_message"]["nodes"][0][ | ||||||
|  |                 "timestamp_precise" | ||||||
|  |             ] | ||||||
|  |         plan = None | ||||||
|  |         if data.get("event_reminders") and data["event_reminders"].get("nodes"): | ||||||
|  |             plan = _plan.Plan._from_graphql(data["event_reminders"]["nodes"][0]) | ||||||
|  |  | ||||||
|  |         return cls( | ||||||
|  |             data["thread_key"]["thread_fbid"], | ||||||
|  |             participants=set( | ||||||
|  |                 [ | ||||||
|  |                     node["messaging_actor"]["id"] | ||||||
|  |                     for node in data["all_participants"]["nodes"] | ||||||
|  |                 ] | ||||||
|  |             ), | ||||||
|  |             nicknames=c_info.get("nicknames"), | ||||||
|  |             color=c_info.get("color"), | ||||||
|  |             emoji=c_info.get("emoji"), | ||||||
|  |             admins=set([node.get("id") for node in data.get("thread_admins")]), | ||||||
|  |             approval_mode=bool(data.get("approval_mode")) | ||||||
|  |             if data.get("approval_mode") is not None | ||||||
|  |             else None, | ||||||
|  |             approval_requests=set( | ||||||
|  |                 node["requester"]["id"] | ||||||
|  |                 for node in data["group_approval_queue"]["nodes"] | ||||||
|  |             ) | ||||||
|  |             if data.get("group_approval_queue") | ||||||
|  |             else None, | ||||||
|  |             join_link=data["joinable_mode"].get("link"), | ||||||
|  |             photo=data["image"].get("uri"), | ||||||
|  |             name=data.get("name"), | ||||||
|  |             message_count=data.get("messages_count"), | ||||||
|  |             last_message_timestamp=last_message_timestamp, | ||||||
|  |             plan=plan, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def _to_send_data(self): | ||||||
|  |         return {"thread_fbid": self.uid} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class Room(Group): | ||||||
|  |     """Deprecated. Use `Group` instead.""" | ||||||
|  |  | ||||||
|  |     # True is room is not discoverable | ||||||
|  |     privacy_mode = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def __init__(self, uid, privacy_mode=None, **kwargs): | ||||||
|  |         super(Room, self).__init__(uid, **kwargs) | ||||||
|  |         self.type = ThreadType.ROOM | ||||||
|  |         self.privacy_mode = privacy_mode | ||||||
							
								
								
									
										112
									
								
								fbchat/_location.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,112 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | from ._attachment import Attachment | ||||||
|  | from . import _util | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class LocationAttachment(Attachment): | ||||||
|  |     """Represents a user location. | ||||||
|  |  | ||||||
|  |     Latitude and longitude OR address is provided by Facebook. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     #: Latitude of the location | ||||||
|  |     latitude = attr.ib(None) | ||||||
|  |     #: Longitude of the location | ||||||
|  |     longitude = attr.ib(None) | ||||||
|  |     #: URL of image showing the map of the location | ||||||
|  |     image_url = attr.ib(None, init=False) | ||||||
|  |     #: Width of the image | ||||||
|  |     image_width = attr.ib(None, init=False) | ||||||
|  |     #: Height of the image | ||||||
|  |     image_height = attr.ib(None, init=False) | ||||||
|  |     #: URL to Bing maps with the location | ||||||
|  |     url = attr.ib(None, init=False) | ||||||
|  |     # Address of the location | ||||||
|  |     address = attr.ib(None) | ||||||
|  |  | ||||||
|  |     # Put here for backwards compatibility, so that the init argument order is preserved | ||||||
|  |     uid = attr.ib(None) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         url = data.get("url") | ||||||
|  |         address = _util.get_url_parameter(_util.get_url_parameter(url, "u"), "where1") | ||||||
|  |         try: | ||||||
|  |             latitude, longitude = [float(x) for x in address.split(", ")] | ||||||
|  |             address = None | ||||||
|  |         except ValueError: | ||||||
|  |             latitude, longitude = None, None | ||||||
|  |         rtn = cls( | ||||||
|  |             uid=int(data["deduplication_key"]), | ||||||
|  |             latitude=latitude, | ||||||
|  |             longitude=longitude, | ||||||
|  |             address=address, | ||||||
|  |         ) | ||||||
|  |         media = data.get("media") | ||||||
|  |         if media and media.get("image"): | ||||||
|  |             image = media["image"] | ||||||
|  |             rtn.image_url = image.get("uri") | ||||||
|  |             rtn.image_width = image.get("width") | ||||||
|  |             rtn.image_height = image.get("height") | ||||||
|  |         rtn.url = url | ||||||
|  |         return rtn | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class LiveLocationAttachment(LocationAttachment): | ||||||
|  |     """Represents a live user location.""" | ||||||
|  |  | ||||||
|  |     #: Name of the location | ||||||
|  |     name = attr.ib(None) | ||||||
|  |     #: Timestamp when live location expires | ||||||
|  |     expiration_time = attr.ib(None) | ||||||
|  |     #: True if live location is expired | ||||||
|  |     is_expired = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def __init__(self, name=None, expiration_time=None, is_expired=None, **kwargs): | ||||||
|  |         super(LiveLocationAttachment, self).__init__(**kwargs) | ||||||
|  |         self.expiration_time = expiration_time | ||||||
|  |         self.is_expired = is_expired | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_pull(cls, data): | ||||||
|  |         return cls( | ||||||
|  |             uid=data["id"], | ||||||
|  |             latitude=data["coordinate"]["latitude"] / (10 ** 8) | ||||||
|  |             if not data.get("stopReason") | ||||||
|  |             else None, | ||||||
|  |             longitude=data["coordinate"]["longitude"] / (10 ** 8) | ||||||
|  |             if not data.get("stopReason") | ||||||
|  |             else None, | ||||||
|  |             name=data.get("locationTitle"), | ||||||
|  |             expiration_time=data["expirationTime"], | ||||||
|  |             is_expired=bool(data.get("stopReason")), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         target = data["target"] | ||||||
|  |         rtn = cls( | ||||||
|  |             uid=int(target["live_location_id"]), | ||||||
|  |             latitude=target["coordinate"]["latitude"] | ||||||
|  |             if target.get("coordinate") | ||||||
|  |             else None, | ||||||
|  |             longitude=target["coordinate"]["longitude"] | ||||||
|  |             if target.get("coordinate") | ||||||
|  |             else None, | ||||||
|  |             name=data["title_with_entities"]["text"], | ||||||
|  |             expiration_time=target.get("expiration_time"), | ||||||
|  |             is_expired=target.get("is_expired"), | ||||||
|  |         ) | ||||||
|  |         media = data.get("media") | ||||||
|  |         if media and media.get("image"): | ||||||
|  |             image = media["image"] | ||||||
|  |             rtn.image_url = image.get("uri") | ||||||
|  |             rtn.image_width = image.get("width") | ||||||
|  |             rtn.image_height = image.get("height") | ||||||
|  |         rtn.url = data.get("url") | ||||||
|  |         return rtn | ||||||
							
								
								
									
										395
									
								
								fbchat/_message.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,395 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | import json | ||||||
|  | from string import Formatter | ||||||
|  | from . import _util, _attachment, _location, _file, _quick_reply, _sticker | ||||||
|  | from ._core import Enum | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class EmojiSize(Enum): | ||||||
|  |     """Used to specify the size of a sent emoji.""" | ||||||
|  |  | ||||||
|  |     LARGE = "369239383222810" | ||||||
|  |     MEDIUM = "369239343222814" | ||||||
|  |     SMALL = "369239263222822" | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_tags(cls, tags): | ||||||
|  |         string_to_emojisize = { | ||||||
|  |             "large": cls.LARGE, | ||||||
|  |             "medium": cls.MEDIUM, | ||||||
|  |             "small": cls.SMALL, | ||||||
|  |             "l": cls.LARGE, | ||||||
|  |             "m": cls.MEDIUM, | ||||||
|  |             "s": cls.SMALL, | ||||||
|  |         } | ||||||
|  |         for tag in tags or (): | ||||||
|  |             data = tag.split(":", 1) | ||||||
|  |             if len(data) > 1 and data[0] == "hot_emoji_size": | ||||||
|  |                 return string_to_emojisize.get(data[1]) | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MessageReaction(Enum): | ||||||
|  |     """Used to specify a message reaction.""" | ||||||
|  |  | ||||||
|  |     HEART = "❤" | ||||||
|  |     LOVE = "😍" | ||||||
|  |     SMILE = "😆" | ||||||
|  |     WOW = "😮" | ||||||
|  |     SAD = "😢" | ||||||
|  |     ANGRY = "😠" | ||||||
|  |     YES = "👍" | ||||||
|  |     NO = "👎" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class Mention(object): | ||||||
|  |     """Represents a ``@mention``.""" | ||||||
|  |  | ||||||
|  |     #: The thread ID the mention is pointing at | ||||||
|  |     thread_id = attr.ib() | ||||||
|  |     #: The character where the mention starts | ||||||
|  |     offset = attr.ib(0) | ||||||
|  |     #: The length of the mention | ||||||
|  |     length = attr.ib(10) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class Message(object): | ||||||
|  |     """Represents a Facebook message.""" | ||||||
|  |  | ||||||
|  |     #: The actual message | ||||||
|  |     text = attr.ib(None) | ||||||
|  |     #: A list of :class:`Mention` objects | ||||||
|  |     mentions = attr.ib(factory=list, converter=lambda x: [] if x is None else x) | ||||||
|  |     #: A :class:`EmojiSize`. Size of a sent emoji | ||||||
|  |     emoji_size = attr.ib(None) | ||||||
|  |     #: The message ID | ||||||
|  |     uid = attr.ib(None, init=False) | ||||||
|  |     #: ID of the sender | ||||||
|  |     author = attr.ib(None, init=False) | ||||||
|  |     #: Timestamp of when the message was sent | ||||||
|  |     timestamp = attr.ib(None, init=False) | ||||||
|  |     #: Whether the message is read | ||||||
|  |     is_read = attr.ib(None, init=False) | ||||||
|  |     #: A list of people IDs who read the message, works only with :func:`fbchat.Client.fetchThreadMessages` | ||||||
|  |     read_by = attr.ib(factory=list, init=False) | ||||||
|  |     #: A dictionary with user's IDs as keys, and their :class:`MessageReaction` as values | ||||||
|  |     reactions = attr.ib(factory=dict, init=False) | ||||||
|  |     #: A :class:`Sticker` | ||||||
|  |     sticker = attr.ib(None) | ||||||
|  |     #: A list of attachments | ||||||
|  |     attachments = attr.ib(factory=list, converter=lambda x: [] if x is None else x) | ||||||
|  |     #: A list of :class:`QuickReply` | ||||||
|  |     quick_replies = attr.ib(factory=list, converter=lambda x: [] if x is None else x) | ||||||
|  |     #: Whether the message is unsent (deleted for everyone) | ||||||
|  |     unsent = attr.ib(False, init=False) | ||||||
|  |     #: Message ID you want to reply to | ||||||
|  |     reply_to_id = attr.ib(None) | ||||||
|  |     #: Replied message | ||||||
|  |     replied_to = attr.ib(None, init=False) | ||||||
|  |     #: Whether the message was forwarded | ||||||
|  |     forwarded = attr.ib(False, init=False) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def formatMentions(cls, text, *args, **kwargs): | ||||||
|  |         """Like `str.format`, but takes tuples with a thread id and text instead. | ||||||
|  |  | ||||||
|  |         Return a `Message` object, with the formatted string and relevant mentions. | ||||||
|  |  | ||||||
|  |         >>> Message.formatMentions("Hey {!r}! My name is {}", ("1234", "Peter"), ("4321", "Michael")) | ||||||
|  |         <Message (None): "Hey 'Peter'! My name is Michael", mentions=[<Mention 1234: offset=4 length=7>, <Mention 4321: offset=24 length=7>] emoji_size=None attachments=[]> | ||||||
|  |  | ||||||
|  |         >>> Message.formatMentions("Hey {p}! My name is {}", ("1234", "Michael"), p=("4321", "Peter")) | ||||||
|  |         <Message (None): 'Hey Peter! My name is Michael', mentions=[<Mention 4321: offset=4 length=5>, <Mention 1234: offset=22 length=7>] emoji_size=None attachments=[]> | ||||||
|  |         """ | ||||||
|  |         result = "" | ||||||
|  |         mentions = list() | ||||||
|  |         offset = 0 | ||||||
|  |         f = Formatter() | ||||||
|  |         field_names = [field_name[1] for field_name in f.parse(text)] | ||||||
|  |         automatic = "" in field_names | ||||||
|  |         i = 0 | ||||||
|  |  | ||||||
|  |         for (literal_text, field_name, format_spec, conversion) in f.parse(text): | ||||||
|  |             offset += len(literal_text) | ||||||
|  |             result += literal_text | ||||||
|  |  | ||||||
|  |             if field_name is None: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             if field_name == "": | ||||||
|  |                 field_name = str(i) | ||||||
|  |                 i += 1 | ||||||
|  |             elif automatic and field_name.isdigit(): | ||||||
|  |                 raise ValueError( | ||||||
|  |                     "cannot switch from automatic field numbering to manual field specification" | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |             thread_id, name = f.get_field(field_name, args, kwargs)[0] | ||||||
|  |  | ||||||
|  |             if format_spec: | ||||||
|  |                 name = f.format_field(name, format_spec) | ||||||
|  |             if conversion: | ||||||
|  |                 name = f.convert_field(name, conversion) | ||||||
|  |  | ||||||
|  |             result += name | ||||||
|  |             mentions.append( | ||||||
|  |                 Mention(thread_id=thread_id, offset=offset, length=len(name)) | ||||||
|  |             ) | ||||||
|  |             offset += len(name) | ||||||
|  |  | ||||||
|  |         message = cls(text=result, mentions=mentions) | ||||||
|  |         return message | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def _get_forwarded_from_tags(tags): | ||||||
|  |         if tags is None: | ||||||
|  |             return False | ||||||
|  |         return any(map(lambda tag: "forward" in tag or "copy" in tag, tags)) | ||||||
|  |  | ||||||
|  |     def _to_send_data(self): | ||||||
|  |         data = {} | ||||||
|  |  | ||||||
|  |         if self.text or self.sticker or self.emoji_size: | ||||||
|  |             data["action_type"] = "ma-type:user-generated-message" | ||||||
|  |  | ||||||
|  |         if self.text: | ||||||
|  |             data["body"] = self.text | ||||||
|  |  | ||||||
|  |         for i, mention in enumerate(self.mentions): | ||||||
|  |             data["profile_xmd[{}][id]".format(i)] = mention.thread_id | ||||||
|  |             data["profile_xmd[{}][offset]".format(i)] = mention.offset | ||||||
|  |             data["profile_xmd[{}][length]".format(i)] = mention.length | ||||||
|  |             data["profile_xmd[{}][type]".format(i)] = "p" | ||||||
|  |  | ||||||
|  |         if self.emoji_size: | ||||||
|  |             if self.text: | ||||||
|  |                 data["tags[0]"] = "hot_emoji_size:" + self.emoji_size.name.lower() | ||||||
|  |             else: | ||||||
|  |                 data["sticker_id"] = self.emoji_size.value | ||||||
|  |  | ||||||
|  |         if self.sticker: | ||||||
|  |             data["sticker_id"] = self.sticker.uid | ||||||
|  |  | ||||||
|  |         if self.quick_replies: | ||||||
|  |             xmd = {"quick_replies": []} | ||||||
|  |             for quick_reply in self.quick_replies: | ||||||
|  |                 # TODO: Move this to `_quick_reply.py` | ||||||
|  |                 q = dict() | ||||||
|  |                 q["content_type"] = quick_reply._type | ||||||
|  |                 q["payload"] = quick_reply.payload | ||||||
|  |                 q["external_payload"] = quick_reply.external_payload | ||||||
|  |                 q["data"] = quick_reply.data | ||||||
|  |                 if quick_reply.is_response: | ||||||
|  |                     q["ignore_for_webhook"] = False | ||||||
|  |                 if isinstance(quick_reply, _quick_reply.QuickReplyText): | ||||||
|  |                     q["title"] = quick_reply.title | ||||||
|  |                 if not isinstance(quick_reply, _quick_reply.QuickReplyLocation): | ||||||
|  |                     q["image_url"] = quick_reply.image_url | ||||||
|  |                 xmd["quick_replies"].append(q) | ||||||
|  |             if len(self.quick_replies) == 1 and self.quick_replies[0].is_response: | ||||||
|  |                 xmd["quick_replies"] = xmd["quick_replies"][0] | ||||||
|  |             data["platform_xmd"] = json.dumps(xmd) | ||||||
|  |  | ||||||
|  |         if self.reply_to_id: | ||||||
|  |             data["replied_to_message_id"] = self.reply_to_id | ||||||
|  |  | ||||||
|  |         return data | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         if data.get("message_sender") is None: | ||||||
|  |             data["message_sender"] = {} | ||||||
|  |         if data.get("message") is None: | ||||||
|  |             data["message"] = {} | ||||||
|  |         tags = data.get("tags_list") | ||||||
|  |         rtn = cls( | ||||||
|  |             text=data["message"].get("text"), | ||||||
|  |             mentions=[ | ||||||
|  |                 Mention( | ||||||
|  |                     m.get("entity", {}).get("id"), | ||||||
|  |                     offset=m.get("offset"), | ||||||
|  |                     length=m.get("length"), | ||||||
|  |                 ) | ||||||
|  |                 for m in data["message"].get("ranges") or () | ||||||
|  |             ], | ||||||
|  |             emoji_size=EmojiSize._from_tags(tags), | ||||||
|  |             sticker=_sticker.Sticker._from_graphql(data.get("sticker")), | ||||||
|  |         ) | ||||||
|  |         rtn.forwarded = cls._get_forwarded_from_tags(tags) | ||||||
|  |         rtn.uid = str(data["message_id"]) | ||||||
|  |         rtn.author = str(data["message_sender"]["id"]) | ||||||
|  |         rtn.timestamp = data.get("timestamp_precise") | ||||||
|  |         rtn.unsent = False | ||||||
|  |         if data.get("unread") is not None: | ||||||
|  |             rtn.is_read = not data["unread"] | ||||||
|  |         rtn.reactions = { | ||||||
|  |             str(r["user"]["id"]): MessageReaction._extend_if_invalid(r["reaction"]) | ||||||
|  |             for r in data["message_reactions"] | ||||||
|  |         } | ||||||
|  |         if data.get("blob_attachments") is not None: | ||||||
|  |             rtn.attachments = [ | ||||||
|  |                 _file.graphql_to_attachment(attachment) | ||||||
|  |                 for attachment in data["blob_attachments"] | ||||||
|  |             ] | ||||||
|  |         if data.get("platform_xmd_encoded"): | ||||||
|  |             quick_replies = json.loads(data["platform_xmd_encoded"]).get( | ||||||
|  |                 "quick_replies" | ||||||
|  |             ) | ||||||
|  |             if isinstance(quick_replies, list): | ||||||
|  |                 rtn.quick_replies = [ | ||||||
|  |                     _quick_reply.graphql_to_quick_reply(q) for q in quick_replies | ||||||
|  |                 ] | ||||||
|  |             elif isinstance(quick_replies, dict): | ||||||
|  |                 rtn.quick_replies = [ | ||||||
|  |                     _quick_reply.graphql_to_quick_reply(quick_replies, is_response=True) | ||||||
|  |                 ] | ||||||
|  |         if data.get("extensible_attachment") is not None: | ||||||
|  |             attachment = graphql_to_extensible_attachment(data["extensible_attachment"]) | ||||||
|  |             if isinstance(attachment, _attachment.UnsentMessage): | ||||||
|  |                 rtn.unsent = True | ||||||
|  |             elif attachment: | ||||||
|  |                 rtn.attachments.append(attachment) | ||||||
|  |         if data.get("replied_to_message") is not None: | ||||||
|  |             rtn.replied_to = cls._from_graphql(data["replied_to_message"]["message"]) | ||||||
|  |             rtn.reply_to_id = rtn.replied_to.uid | ||||||
|  |         return rtn | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_reply(cls, data): | ||||||
|  |         tags = data["messageMetadata"].get("tags") | ||||||
|  |         rtn = cls( | ||||||
|  |             text=data.get("body"), | ||||||
|  |             mentions=[ | ||||||
|  |                 Mention(m.get("i"), offset=m.get("o"), length=m.get("l")) | ||||||
|  |                 for m in json.loads(data.get("data", {}).get("prng", "[]")) | ||||||
|  |             ], | ||||||
|  |             emoji_size=EmojiSize._from_tags(tags), | ||||||
|  |         ) | ||||||
|  |         metadata = data.get("messageMetadata", {}) | ||||||
|  |         rtn.forwarded = cls._get_forwarded_from_tags(tags) | ||||||
|  |         rtn.uid = metadata.get("messageId") | ||||||
|  |         rtn.author = str(metadata.get("actorFbId")) | ||||||
|  |         rtn.timestamp = metadata.get("timestamp") | ||||||
|  |         rtn.unsent = False | ||||||
|  |         if data.get("data", {}).get("platform_xmd"): | ||||||
|  |             quick_replies = json.loads(data["data"]["platform_xmd"]).get( | ||||||
|  |                 "quick_replies" | ||||||
|  |             ) | ||||||
|  |             if isinstance(quick_replies, list): | ||||||
|  |                 rtn.quick_replies = [ | ||||||
|  |                     _quick_reply.graphql_to_quick_reply(q) for q in quick_replies | ||||||
|  |                 ] | ||||||
|  |             elif isinstance(quick_replies, dict): | ||||||
|  |                 rtn.quick_replies = [ | ||||||
|  |                     _quick_reply.graphql_to_quick_reply(quick_replies, is_response=True) | ||||||
|  |                 ] | ||||||
|  |         if data.get("attachments") is not None: | ||||||
|  |             for attachment in data["attachments"]: | ||||||
|  |                 attachment = json.loads(attachment["mercuryJSON"]) | ||||||
|  |                 if attachment.get("blob_attachment"): | ||||||
|  |                     rtn.attachments.append( | ||||||
|  |                         _file.graphql_to_attachment(attachment["blob_attachment"]) | ||||||
|  |                     ) | ||||||
|  |                 if attachment.get("extensible_attachment"): | ||||||
|  |                     extensible_attachment = graphql_to_extensible_attachment( | ||||||
|  |                         attachment["extensible_attachment"] | ||||||
|  |                     ) | ||||||
|  |                     if isinstance(extensible_attachment, _attachment.UnsentMessage): | ||||||
|  |                         rtn.unsent = True | ||||||
|  |                     else: | ||||||
|  |                         rtn.attachments.append(extensible_attachment) | ||||||
|  |                 if attachment.get("sticker_attachment"): | ||||||
|  |                     rtn.sticker = _sticker.Sticker._from_graphql( | ||||||
|  |                         attachment["sticker_attachment"] | ||||||
|  |                     ) | ||||||
|  |         return rtn | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_pull(cls, data, mid=None, tags=None, author=None, timestamp=None): | ||||||
|  |         rtn = cls(text=data.get("body")) | ||||||
|  |         rtn.uid = mid | ||||||
|  |         rtn.author = author | ||||||
|  |         rtn.timestamp = timestamp | ||||||
|  |  | ||||||
|  |         if data.get("data") and data["data"].get("prng"): | ||||||
|  |             try: | ||||||
|  |                 rtn.mentions = [ | ||||||
|  |                     Mention( | ||||||
|  |                         str(mention.get("i")), | ||||||
|  |                         offset=mention.get("o"), | ||||||
|  |                         length=mention.get("l"), | ||||||
|  |                     ) | ||||||
|  |                     for mention in _util.parse_json(data["data"]["prng"]) | ||||||
|  |                 ] | ||||||
|  |             except Exception: | ||||||
|  |                 _util.log.exception("An exception occured while reading attachments") | ||||||
|  |  | ||||||
|  |         if data.get("attachments"): | ||||||
|  |             try: | ||||||
|  |                 for a in data["attachments"]: | ||||||
|  |                     mercury = a["mercury"] | ||||||
|  |                     if mercury.get("blob_attachment"): | ||||||
|  |                         image_metadata = a.get("imageMetadata", {}) | ||||||
|  |                         attach_type = mercury["blob_attachment"]["__typename"] | ||||||
|  |                         attachment = _file.graphql_to_attachment( | ||||||
|  |                             mercury["blob_attachment"] | ||||||
|  |                         ) | ||||||
|  |  | ||||||
|  |                         if attach_type in [ | ||||||
|  |                             "MessageFile", | ||||||
|  |                             "MessageVideo", | ||||||
|  |                             "MessageAudio", | ||||||
|  |                         ]: | ||||||
|  |                             # TODO: Add more data here for audio files | ||||||
|  |                             attachment.size = int(a["fileSize"]) | ||||||
|  |                         rtn.attachments.append(attachment) | ||||||
|  |  | ||||||
|  |                     elif mercury.get("sticker_attachment"): | ||||||
|  |                         rtn.sticker = _sticker.Sticker._from_graphql( | ||||||
|  |                             mercury["sticker_attachment"] | ||||||
|  |                         ) | ||||||
|  |  | ||||||
|  |                     elif mercury.get("extensible_attachment"): | ||||||
|  |                         attachment = graphql_to_extensible_attachment( | ||||||
|  |                             mercury["extensible_attachment"] | ||||||
|  |                         ) | ||||||
|  |                         if isinstance(attachment, _attachment.UnsentMessage): | ||||||
|  |                             rtn.unsent = True | ||||||
|  |                         elif attachment: | ||||||
|  |                             rtn.attachments.append(attachment) | ||||||
|  |  | ||||||
|  |             except Exception: | ||||||
|  |                 _util.log.exception( | ||||||
|  |                     "An exception occured while reading attachments: {}".format( | ||||||
|  |                         data["attachments"] | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |         rtn.emoji_size = EmojiSize._from_tags(tags) | ||||||
|  |         rtn.forwarded = cls._get_forwarded_from_tags(tags) | ||||||
|  |         return rtn | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def graphql_to_extensible_attachment(data): | ||||||
|  |     story = data.get("story_attachment") | ||||||
|  |     if not story: | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     target = story.get("target") | ||||||
|  |     if not target: | ||||||
|  |         return _attachment.UnsentMessage(uid=data.get("legacy_attachment_id")) | ||||||
|  |  | ||||||
|  |     _type = target["__typename"] | ||||||
|  |     if _type == "MessageLocation": | ||||||
|  |         return _location.LocationAttachment._from_graphql(story) | ||||||
|  |     elif _type == "MessageLiveLocation": | ||||||
|  |         return _location.LiveLocationAttachment._from_graphql(story) | ||||||
|  |     elif _type in ["ExternalUrl", "Story"]: | ||||||
|  |         return _attachment.ShareAttachment._from_graphql(story) | ||||||
|  |  | ||||||
|  |     return None | ||||||
							
								
								
									
										339
									
								
								fbchat/_mqtt.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,339 @@ | |||||||
|  | import attr | ||||||
|  | import random | ||||||
|  | import paho.mqtt.client | ||||||
|  | from ._core import log | ||||||
|  | from . import _util, _exception, _graphql | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def generate_session_id(): | ||||||
|  |     """Generate a random session ID between 1 and 9007199254740991.""" | ||||||
|  |     return random.randint(1, 2 ** 53) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(slots=True) | ||||||
|  | class Mqtt(object): | ||||||
|  |     _state = attr.ib() | ||||||
|  |     _mqtt = attr.ib() | ||||||
|  |     _on_message = attr.ib() | ||||||
|  |     _chat_on = attr.ib() | ||||||
|  |     _foreground = attr.ib() | ||||||
|  |     _sequence_id = attr.ib() | ||||||
|  |     _sync_token = attr.ib(None) | ||||||
|  |  | ||||||
|  |     _HOST = "edge-chat.facebook.com" | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def connect(cls, state, on_message, chat_on, foreground): | ||||||
|  |         mqtt = paho.mqtt.client.Client( | ||||||
|  |             client_id="mqttwsclient", | ||||||
|  |             clean_session=True, | ||||||
|  |             protocol=paho.mqtt.client.MQTTv31, | ||||||
|  |             transport="websockets", | ||||||
|  |         ) | ||||||
|  |         mqtt.enable_logger() | ||||||
|  |         # mqtt.max_inflight_messages_set(20)  # The rest will get queued | ||||||
|  |         # mqtt.max_queued_messages_set(0)  # Unlimited messages can be queued | ||||||
|  |         # mqtt.message_retry_set(20)  # Retry sending for at least 20 seconds | ||||||
|  |         # mqtt.reconnect_delay_set(min_delay=1, max_delay=120) | ||||||
|  |         # TODO: Is region (lla | atn | odn | others?) important? | ||||||
|  |         mqtt.tls_set() | ||||||
|  |  | ||||||
|  |         self = cls( | ||||||
|  |             state=state, | ||||||
|  |             mqtt=mqtt, | ||||||
|  |             on_message=on_message, | ||||||
|  |             chat_on=chat_on, | ||||||
|  |             foreground=foreground, | ||||||
|  |             sequence_id=cls._fetch_sequence_id(state), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         # Configure callbacks | ||||||
|  |         mqtt.on_message = self._on_message_handler | ||||||
|  |         mqtt.on_connect = self._on_connect_handler | ||||||
|  |  | ||||||
|  |         self._configure_connect_options() | ||||||
|  |  | ||||||
|  |         # Attempt to connect | ||||||
|  |         try: | ||||||
|  |             rc = mqtt.connect(self._HOST, 443, keepalive=10) | ||||||
|  |         except ( | ||||||
|  |             # Taken from .loop_forever | ||||||
|  |             paho.mqtt.client.socket.error, | ||||||
|  |             OSError, | ||||||
|  |             paho.mqtt.client.WebsocketConnectionError, | ||||||
|  |         ) as e: | ||||||
|  |             raise _exception.FBchatException("MQTT connection failed") | ||||||
|  |  | ||||||
|  |         # Raise error if connecting failed | ||||||
|  |         if rc != paho.mqtt.client.MQTT_ERR_SUCCESS: | ||||||
|  |             err = paho.mqtt.client.error_string(rc) | ||||||
|  |             raise _exception.FBchatException("MQTT connection failed: {}".format(err)) | ||||||
|  |  | ||||||
|  |         return self | ||||||
|  |  | ||||||
|  |     def _on_message_handler(self, client, userdata, message): | ||||||
|  |         # Parse payload JSON | ||||||
|  |         try: | ||||||
|  |             j = _util.parse_json(message.payload.decode("utf-8")) | ||||||
|  |         except (_exception.FBchatFacebookError, UnicodeDecodeError): | ||||||
|  |             log.exception("Failed parsing MQTT data on %s as JSON", message.topic) | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         log.debug("MQTT payload: %s, %s", message.topic, j) | ||||||
|  |  | ||||||
|  |         if message.topic == "/t_ms": | ||||||
|  |             # Update sync_token when received | ||||||
|  |             # This is received in the first message after we've created a messenger | ||||||
|  |             # sync queue. | ||||||
|  |             if "syncToken" in j and "firstDeltaSeqId" in j: | ||||||
|  |                 self._sync_token = j["syncToken"] | ||||||
|  |                 self._sequence_id = j["firstDeltaSeqId"] | ||||||
|  |                 return | ||||||
|  |  | ||||||
|  |             # Update last sequence id when received | ||||||
|  |             if "lastIssuedSeqId" in j: | ||||||
|  |                 self._sequence_id = j["lastIssuedSeqId"] | ||||||
|  |  | ||||||
|  |             if "errorCode" in j: | ||||||
|  |                 error = j["errorCode"] | ||||||
|  |                 # TODO: 'F\xfa\x84\x8c\x85\xf8\xbc-\x88 FB_PAGES_INSUFFICIENT_PERMISSION\x00' | ||||||
|  |                 if error in ("ERROR_QUEUE_NOT_FOUND", "ERROR_QUEUE_OVERFLOW"): | ||||||
|  |                     # ERROR_QUEUE_NOT_FOUND means that the queue was deleted, since too | ||||||
|  |                     # much time passed, or that it was simply missing | ||||||
|  |                     # ERROR_QUEUE_OVERFLOW means that the sequence id was too small, so | ||||||
|  |                     # the desired events could not be retrieved | ||||||
|  |                     log.error( | ||||||
|  |                         "The MQTT listener was disconnected for too long," | ||||||
|  |                         " events may have been lost" | ||||||
|  |                     ) | ||||||
|  |                     self._sync_token = None | ||||||
|  |                     self._sequence_id = self._fetch_sequence_id(self._state) | ||||||
|  |                     self._messenger_queue_publish() | ||||||
|  |                     # TODO: Signal to the user that they should reload their data! | ||||||
|  |                     return | ||||||
|  |                 log.error("MQTT error code %s received", error) | ||||||
|  |                 return | ||||||
|  |  | ||||||
|  |         # Call the external callback | ||||||
|  |         self._on_message(message.topic, j) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def _fetch_sequence_id(state): | ||||||
|  |         """Fetch sequence ID.""" | ||||||
|  |         params = { | ||||||
|  |             "limit": 1, | ||||||
|  |             "tags": ["INBOX"], | ||||||
|  |             "before": None, | ||||||
|  |             "includeDeliveryReceipts": False, | ||||||
|  |             "includeSeqID": True, | ||||||
|  |         } | ||||||
|  |         log.debug("Fetching MQTT sequence ID") | ||||||
|  |         # Same request as in `Client.fetchThreadList` | ||||||
|  |         (j,) = state._graphql_requests(_graphql.from_doc_id("1349387578499440", params)) | ||||||
|  |         sequence_id = j["viewer"]["message_threads"]["sync_sequence_id"] | ||||||
|  |         if not sequence_id: | ||||||
|  |             raise _exception.FBchatNotLoggedIn("Failed fetching sequence id") | ||||||
|  |         return int(sequence_id) | ||||||
|  |  | ||||||
|  |     def _on_connect_handler(self, client, userdata, flags, rc): | ||||||
|  |         if rc == 21: | ||||||
|  |             raise _exception.FBchatException( | ||||||
|  |                 "Failed connecting. Maybe your cookies are wrong?" | ||||||
|  |             ) | ||||||
|  |         if rc != 0: | ||||||
|  |             return  # Don't try to send publish if the connection failed | ||||||
|  |  | ||||||
|  |         self._messenger_queue_publish() | ||||||
|  |  | ||||||
|  |     def _messenger_queue_publish(self): | ||||||
|  |         # configure receiving messages. | ||||||
|  |         payload = { | ||||||
|  |             "sync_api_version": 10, | ||||||
|  |             "max_deltas_able_to_process": 1000, | ||||||
|  |             "delta_batch_size": 500, | ||||||
|  |             "encoding": "JSON", | ||||||
|  |             "entity_fbid": self._state.user_id, | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         # If we don't have a sync_token, create a new messenger queue | ||||||
|  |         # This is done so that across reconnects, if we've received a sync token, we | ||||||
|  |         # SHOULD receive a piece of data in /t_ms exactly once! | ||||||
|  |         if self._sync_token is None: | ||||||
|  |             topic = "/messenger_sync_create_queue" | ||||||
|  |             payload["initial_titan_sequence_id"] = str(self._sequence_id) | ||||||
|  |             payload["device_params"] = None | ||||||
|  |         else: | ||||||
|  |             topic = "/messenger_sync_get_diffs" | ||||||
|  |             payload["last_seq_id"] = str(self._sequence_id) | ||||||
|  |             payload["sync_token"] = self._sync_token | ||||||
|  |  | ||||||
|  |         self._mqtt.publish(topic, _util.json_minimal(payload), qos=1) | ||||||
|  |  | ||||||
|  |     def _configure_connect_options(self): | ||||||
|  |         # Generate a new session ID on each reconnect | ||||||
|  |         session_id = generate_session_id() | ||||||
|  |  | ||||||
|  |         topics = [ | ||||||
|  |             # Things that happen in chats (e.g. messages) | ||||||
|  |             "/t_ms", | ||||||
|  |             # Group typing notifications | ||||||
|  |             "/thread_typing", | ||||||
|  |             # Private chat typing notifications | ||||||
|  |             "/orca_typing_notifications", | ||||||
|  |             # Active notifications | ||||||
|  |             "/orca_presence", | ||||||
|  |             # Other notifications not related to chats (e.g. friend requests) | ||||||
|  |             "/legacy_web", | ||||||
|  |             # Facebook's continuous error reporting/logging? | ||||||
|  |             "/br_sr", | ||||||
|  |             # Response to /br_sr | ||||||
|  |             "/sr_res", | ||||||
|  |             # Data about user-to-user calls | ||||||
|  |             # TODO: Investigate the response from this! (A bunch of binary data) | ||||||
|  |             # "/t_rtc", | ||||||
|  |             # TODO: Find out what this does! | ||||||
|  |             # TODO: Investigate the response from this! (A bunch of binary data) | ||||||
|  |             # "/t_p", | ||||||
|  |             # TODO: Find out what this does! | ||||||
|  |             "/webrtc", | ||||||
|  |             # TODO: Find out what this does! | ||||||
|  |             "/onevc", | ||||||
|  |             # TODO: Find out what this does! | ||||||
|  |             "/notify_disconnect", | ||||||
|  |             # Old, no longer active topics | ||||||
|  |             # These are here just in case something interesting pops up | ||||||
|  |             "/inbox", | ||||||
|  |             "/mercury", | ||||||
|  |             "/messaging_events", | ||||||
|  |             "/orca_message_notifications", | ||||||
|  |             "/pp", | ||||||
|  |             "/webrtc_response", | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |         username = { | ||||||
|  |             # The user ID | ||||||
|  |             "u": self._state.user_id, | ||||||
|  |             # Session ID | ||||||
|  |             "s": session_id, | ||||||
|  |             # Active status setting | ||||||
|  |             "chat_on": self._chat_on, | ||||||
|  |             # foreground_state - Whether the window is focused | ||||||
|  |             "fg": self._foreground, | ||||||
|  |             # Can be any random ID | ||||||
|  |             "d": self._state._client_id, | ||||||
|  |             # Application ID, taken from facebook.com | ||||||
|  |             "aid": 219994525426954, | ||||||
|  |             # MQTT extension by FB, allows making a SUBSCRIBE while CONNECTing | ||||||
|  |             "st": topics, | ||||||
|  |             # MQTT extension by FB, allows making a PUBLISH while CONNECTing | ||||||
|  |             # Using this is more efficient, but the same can be acheived with: | ||||||
|  |             #     def on_connect(*args): | ||||||
|  |             #         mqtt.publish(topic, payload, qos=1) | ||||||
|  |             #     mqtt.on_connect = on_connect | ||||||
|  |             # TODO: For some reason this doesn't work! | ||||||
|  |             "pm": [ | ||||||
|  |                 # { | ||||||
|  |                 #     "topic": topic, | ||||||
|  |                 #     "payload": payload, | ||||||
|  |                 #     "qos": 1, | ||||||
|  |                 #     "messageId": 65536, | ||||||
|  |                 # } | ||||||
|  |             ], | ||||||
|  |             # Unknown parameters | ||||||
|  |             "cp": 3, | ||||||
|  |             "ecp": 10, | ||||||
|  |             "ct": "websocket", | ||||||
|  |             "mqtt_sid": "", | ||||||
|  |             "dc": "", | ||||||
|  |             "no_auto_fg": True, | ||||||
|  |             "gas": None, | ||||||
|  |             "pack": [], | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         # TODO: Make this thread safe | ||||||
|  |         self._mqtt.username_pw_set(_util.json_minimal(username)) | ||||||
|  |  | ||||||
|  |         headers = { | ||||||
|  |             # TODO: Make this access thread safe | ||||||
|  |             "Cookie": _util.get_cookie_header( | ||||||
|  |                 self._state._session, "https://edge-chat.facebook.com/chat" | ||||||
|  |             ), | ||||||
|  |             "User-Agent": self._state._session.headers["User-Agent"], | ||||||
|  |             "Origin": "https://www.facebook.com", | ||||||
|  |             "Host": self._HOST, | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         self._mqtt.ws_set_options( | ||||||
|  |             path="/chat?sid={}".format(session_id), headers=headers | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def loop_once(self, on_error=None): | ||||||
|  |         """Run the listening loop once. | ||||||
|  |  | ||||||
|  |         Returns whether to keep listening or not. | ||||||
|  |         """ | ||||||
|  |         rc = self._mqtt.loop(timeout=1.0) | ||||||
|  |  | ||||||
|  |         # If disconnect() has been called | ||||||
|  |         if self._mqtt._state == paho.mqtt.client.mqtt_cs_disconnecting: | ||||||
|  |             return False  # Stop listening | ||||||
|  |  | ||||||
|  |         if rc != paho.mqtt.client.MQTT_ERR_SUCCESS: | ||||||
|  |             # If known/expected error | ||||||
|  |             if rc == paho.mqtt.client.MQTT_ERR_CONN_LOST: | ||||||
|  |                 log.warning("Connection lost, retrying") | ||||||
|  |             elif rc == paho.mqtt.client.MQTT_ERR_NOMEM: | ||||||
|  |                 # This error is wrongly classified | ||||||
|  |                 # See https://github.com/eclipse/paho.mqtt.python/issues/340 | ||||||
|  |                 log.warning("Connection error, retrying") | ||||||
|  |             elif rc == paho.mqtt.client.MQTT_ERR_CONN_REFUSED: | ||||||
|  |                 raise _exception.FBchatNotLoggedIn("MQTT connection refused") | ||||||
|  |             else: | ||||||
|  |                 err = paho.mqtt.client.error_string(rc) | ||||||
|  |                 log.error("MQTT Error: %s", err) | ||||||
|  |                 # For backwards compatibility | ||||||
|  |                 if on_error: | ||||||
|  |                     on_error(_exception.FBchatException("MQTT Error {}".format(err))) | ||||||
|  |  | ||||||
|  |             # Wait before reconnecting | ||||||
|  |             self._mqtt._reconnect_wait() | ||||||
|  |  | ||||||
|  |             # Try reconnecting | ||||||
|  |             self._configure_connect_options() | ||||||
|  |             try: | ||||||
|  |                 self._mqtt.reconnect() | ||||||
|  |             except ( | ||||||
|  |                 # Taken from .loop_forever | ||||||
|  |                 paho.mqtt.client.socket.error, | ||||||
|  |                 OSError, | ||||||
|  |                 paho.mqtt.client.WebsocketConnectionError, | ||||||
|  |             ) as e: | ||||||
|  |                 log.debug("MQTT reconnection failed: %s", e) | ||||||
|  |  | ||||||
|  |         return True  # Keep listening | ||||||
|  |  | ||||||
|  |     def disconnect(self): | ||||||
|  |         self._mqtt.disconnect() | ||||||
|  |  | ||||||
|  |     def set_foreground(self, value): | ||||||
|  |         payload = _util.json_minimal({"foreground": value}) | ||||||
|  |         info = self._mqtt.publish("/foreground_state", payload=payload, qos=1) | ||||||
|  |         self._foreground = value | ||||||
|  |         # TODO: We can't wait for this, since the loop is running with .loop_forever() | ||||||
|  |         # info.wait_for_publish() | ||||||
|  |  | ||||||
|  |     def set_chat_on(self, value): | ||||||
|  |         # TODO: Is this the right request to make? | ||||||
|  |         data = {"make_user_available_when_in_foreground": value} | ||||||
|  |         payload = _util.json_minimal(data) | ||||||
|  |         info = self._mqtt.publish("/set_client_settings", payload=payload, qos=1) | ||||||
|  |         self._chat_on = value | ||||||
|  |         # TODO: We can't wait for this, since the loop is running with .loop_forever() | ||||||
|  |         # info.wait_for_publish() | ||||||
|  |  | ||||||
|  |     # def send_additional_contacts(self, additional_contacts): | ||||||
|  |     #     payload = _util.json_minimal({"additional_contacts": additional_contacts}) | ||||||
|  |     #     info = self._mqtt.publish("/send_additional_contacts", payload=payload, qos=1) | ||||||
|  |     # | ||||||
|  |     # def browser_close(self): | ||||||
|  |     #     info = self._mqtt.publish("/browser_close", payload=b"{}", qos=1) | ||||||
							
								
								
									
										60
									
								
								fbchat/_page.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,60 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | from . import _plan | ||||||
|  | from ._thread import ThreadType, Thread | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class Page(Thread): | ||||||
|  |     """Represents a Facebook page. Inherits `Thread`.""" | ||||||
|  |  | ||||||
|  |     #: The page's custom URL | ||||||
|  |     url = attr.ib(None) | ||||||
|  |     #: The name of the page's location city | ||||||
|  |     city = attr.ib(None) | ||||||
|  |     #: Amount of likes the page has | ||||||
|  |     likes = attr.ib(None) | ||||||
|  |     #: Some extra information about the page | ||||||
|  |     sub_title = attr.ib(None) | ||||||
|  |     #: The page's category | ||||||
|  |     category = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         uid, | ||||||
|  |         url=None, | ||||||
|  |         city=None, | ||||||
|  |         likes=None, | ||||||
|  |         sub_title=None, | ||||||
|  |         category=None, | ||||||
|  |         **kwargs | ||||||
|  |     ): | ||||||
|  |         super(Page, self).__init__(ThreadType.PAGE, uid, **kwargs) | ||||||
|  |         self.url = url | ||||||
|  |         self.city = city | ||||||
|  |         self.likes = likes | ||||||
|  |         self.sub_title = sub_title | ||||||
|  |         self.category = category | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         if data.get("profile_picture") is None: | ||||||
|  |             data["profile_picture"] = {} | ||||||
|  |         if data.get("city") is None: | ||||||
|  |             data["city"] = {} | ||||||
|  |         plan = None | ||||||
|  |         if data.get("event_reminders") and data["event_reminders"].get("nodes"): | ||||||
|  |             plan = _plan.Plan._from_graphql(data["event_reminders"]["nodes"][0]) | ||||||
|  |  | ||||||
|  |         return cls( | ||||||
|  |             data["id"], | ||||||
|  |             url=data.get("url"), | ||||||
|  |             city=data.get("city").get("name"), | ||||||
|  |             category=data.get("category_type"), | ||||||
|  |             photo=data["profile_picture"].get("uri"), | ||||||
|  |             name=data.get("name"), | ||||||
|  |             message_count=data.get("messages_count"), | ||||||
|  |             plan=plan, | ||||||
|  |         ) | ||||||
							
								
								
									
										103
									
								
								fbchat/_plan.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,103 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | import json | ||||||
|  | from ._core import Enum | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GuestStatus(Enum): | ||||||
|  |     INVITED = 1 | ||||||
|  |     GOING = 2 | ||||||
|  |     DECLINED = 3 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class Plan(object): | ||||||
|  |     """Represents a plan.""" | ||||||
|  |  | ||||||
|  |     #: ID of the plan | ||||||
|  |     uid = attr.ib(None, init=False) | ||||||
|  |     #: Plan time (timestamp), only precise down to the minute | ||||||
|  |     time = attr.ib(converter=int) | ||||||
|  |     #: Plan title | ||||||
|  |     title = attr.ib() | ||||||
|  |     #: Plan location name | ||||||
|  |     location = attr.ib(None, converter=lambda x: x or "") | ||||||
|  |     #: Plan location ID | ||||||
|  |     location_id = attr.ib(None, converter=lambda x: x or "") | ||||||
|  |     #: ID of the plan creator | ||||||
|  |     author_id = attr.ib(None, init=False) | ||||||
|  |     #: Dictionary of `User` IDs mapped to their `GuestStatus` | ||||||
|  |     guests = attr.ib(None, init=False) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def going(self): | ||||||
|  |         """List of the `User` IDs who will take part in the plan.""" | ||||||
|  |         return [ | ||||||
|  |             id_ | ||||||
|  |             for id_, status in (self.guests or {}).items() | ||||||
|  |             if status is GuestStatus.GOING | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def declined(self): | ||||||
|  |         """List of the `User` IDs who won't take part in the plan.""" | ||||||
|  |         return [ | ||||||
|  |             id_ | ||||||
|  |             for id_, status in (self.guests or {}).items() | ||||||
|  |             if status is GuestStatus.DECLINED | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def invited(self): | ||||||
|  |         """List of the `User` IDs who are invited to the plan.""" | ||||||
|  |         return [ | ||||||
|  |             id_ | ||||||
|  |             for id_, status in (self.guests or {}).items() | ||||||
|  |             if status is GuestStatus.INVITED | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_pull(cls, data): | ||||||
|  |         rtn = cls( | ||||||
|  |             time=data.get("event_time"), | ||||||
|  |             title=data.get("event_title"), | ||||||
|  |             location=data.get("event_location_name"), | ||||||
|  |             location_id=data.get("event_location_id"), | ||||||
|  |         ) | ||||||
|  |         rtn.uid = data.get("event_id") | ||||||
|  |         rtn.author_id = data.get("event_creator_id") | ||||||
|  |         rtn.guests = { | ||||||
|  |             x["node"]["id"]: GuestStatus[x["guest_list_state"]] | ||||||
|  |             for x in json.loads(data["guest_state_list"]) | ||||||
|  |         } | ||||||
|  |         return rtn | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_fetch(cls, data): | ||||||
|  |         rtn = cls( | ||||||
|  |             time=data.get("event_time"), | ||||||
|  |             title=data.get("title"), | ||||||
|  |             location=data.get("location_name"), | ||||||
|  |             location_id=str(data["location_id"]) if data.get("location_id") else None, | ||||||
|  |         ) | ||||||
|  |         rtn.uid = data.get("oid") | ||||||
|  |         rtn.author_id = data.get("creator_id") | ||||||
|  |         rtn.guests = {id_: GuestStatus[s] for id_, s in data["event_members"].items()} | ||||||
|  |         return rtn | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         rtn = cls( | ||||||
|  |             time=data.get("time"), | ||||||
|  |             title=data.get("event_title"), | ||||||
|  |             location=data.get("location_name"), | ||||||
|  |         ) | ||||||
|  |         rtn.uid = data.get("id") | ||||||
|  |         rtn.author_id = data["lightweight_event_creator"].get("id") | ||||||
|  |         rtn.guests = { | ||||||
|  |             x["node"]["id"]: GuestStatus[x["guest_list_state"]] | ||||||
|  |             for x in data["event_reminder_members"]["edges"] | ||||||
|  |         } | ||||||
|  |         return rtn | ||||||
							
								
								
									
										67
									
								
								fbchat/_poll.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,67 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class Poll(object): | ||||||
|  |     """Represents a poll.""" | ||||||
|  |  | ||||||
|  |     #: Title of the poll | ||||||
|  |     title = attr.ib() | ||||||
|  |     #: List of :class:`PollOption`, can be fetched with :func:`fbchat.Client.fetchPollOptions` | ||||||
|  |     options = attr.ib() | ||||||
|  |     #: Options count | ||||||
|  |     options_count = attr.ib(None) | ||||||
|  |     #: ID of the poll | ||||||
|  |     uid = attr.ib(None) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         return cls( | ||||||
|  |             uid=int(data["id"]), | ||||||
|  |             title=data.get("title") if data.get("title") else data.get("text"), | ||||||
|  |             options=[PollOption._from_graphql(m) for m in data.get("options")], | ||||||
|  |             options_count=data.get("total_count"), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class PollOption(object): | ||||||
|  |     """Represents a poll option.""" | ||||||
|  |  | ||||||
|  |     #: Text of the poll option | ||||||
|  |     text = attr.ib() | ||||||
|  |     #: Whether vote when creating or client voted | ||||||
|  |     vote = attr.ib(False) | ||||||
|  |     #: ID of the users who voted for this poll option | ||||||
|  |     voters = attr.ib(None) | ||||||
|  |     #: Votes count | ||||||
|  |     votes_count = attr.ib(None) | ||||||
|  |     #: ID of the poll option | ||||||
|  |     uid = attr.ib(None) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         if data.get("viewer_has_voted") is None: | ||||||
|  |             vote = None | ||||||
|  |         elif isinstance(data["viewer_has_voted"], bool): | ||||||
|  |             vote = data["viewer_has_voted"] | ||||||
|  |         else: | ||||||
|  |             vote = data["viewer_has_voted"] == "true" | ||||||
|  |         return cls( | ||||||
|  |             uid=int(data["id"]), | ||||||
|  |             text=data.get("text"), | ||||||
|  |             vote=vote, | ||||||
|  |             voters=( | ||||||
|  |                 [m.get("node").get("id") for m in data.get("voters").get("edges")] | ||||||
|  |                 if isinstance(data.get("voters"), dict) | ||||||
|  |                 else data.get("voters") | ||||||
|  |             ), | ||||||
|  |             votes_count=( | ||||||
|  |                 data.get("voters").get("count") | ||||||
|  |                 if isinstance(data.get("voters"), dict) | ||||||
|  |                 else data.get("total_count") | ||||||
|  |             ), | ||||||
|  |         ) | ||||||
							
								
								
									
										99
									
								
								fbchat/_quick_reply.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,99 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | from ._attachment import Attachment | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class QuickReply(object): | ||||||
|  |     """Represents a quick reply.""" | ||||||
|  |  | ||||||
|  |     #: Payload of the quick reply | ||||||
|  |     payload = attr.ib(None) | ||||||
|  |     #: External payload for responses | ||||||
|  |     external_payload = attr.ib(None, init=False) | ||||||
|  |     #: Additional data | ||||||
|  |     data = attr.ib(None) | ||||||
|  |     #: Whether it's a response for a quick reply | ||||||
|  |     is_response = attr.ib(False) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class QuickReplyText(QuickReply): | ||||||
|  |     """Represents a text quick reply.""" | ||||||
|  |  | ||||||
|  |     #: Title of the quick reply | ||||||
|  |     title = attr.ib(None) | ||||||
|  |     #: URL of the quick reply image (optional) | ||||||
|  |     image_url = attr.ib(None) | ||||||
|  |     #: Type of the quick reply | ||||||
|  |     _type = "text" | ||||||
|  |  | ||||||
|  |     def __init__(self, title=None, image_url=None, **kwargs): | ||||||
|  |         super(QuickReplyText, self).__init__(**kwargs) | ||||||
|  |         self.title = title | ||||||
|  |         self.image_url = image_url | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class QuickReplyLocation(QuickReply): | ||||||
|  |     """Represents a location quick reply (Doesn't work on mobile).""" | ||||||
|  |  | ||||||
|  |     #: Type of the quick reply | ||||||
|  |     _type = "location" | ||||||
|  |  | ||||||
|  |     def __init__(self, **kwargs): | ||||||
|  |         super(QuickReplyLocation, self).__init__(**kwargs) | ||||||
|  |         self.is_response = False | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class QuickReplyPhoneNumber(QuickReply): | ||||||
|  |     """Represents a phone number quick reply (Doesn't work on mobile).""" | ||||||
|  |  | ||||||
|  |     #: URL of the quick reply image (optional) | ||||||
|  |     image_url = attr.ib(None) | ||||||
|  |     #: Type of the quick reply | ||||||
|  |     _type = "user_phone_number" | ||||||
|  |  | ||||||
|  |     def __init__(self, image_url=None, **kwargs): | ||||||
|  |         super(QuickReplyPhoneNumber, self).__init__(**kwargs) | ||||||
|  |         self.image_url = image_url | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class QuickReplyEmail(QuickReply): | ||||||
|  |     """Represents an email quick reply (Doesn't work on mobile).""" | ||||||
|  |  | ||||||
|  |     #: URL of the quick reply image (optional) | ||||||
|  |     image_url = attr.ib(None) | ||||||
|  |     #: Type of the quick reply | ||||||
|  |     _type = "user_email" | ||||||
|  |  | ||||||
|  |     def __init__(self, image_url=None, **kwargs): | ||||||
|  |         super(QuickReplyEmail, self).__init__(**kwargs) | ||||||
|  |         self.image_url = image_url | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def graphql_to_quick_reply(q, is_response=False): | ||||||
|  |     data = dict() | ||||||
|  |     _type = q.get("content_type").lower() | ||||||
|  |     if q.get("payload"): | ||||||
|  |         data["payload"] = q["payload"] | ||||||
|  |     if q.get("data"): | ||||||
|  |         data["data"] = q["data"] | ||||||
|  |     if q.get("image_url") and _type is not QuickReplyLocation._type: | ||||||
|  |         data["image_url"] = q["image_url"] | ||||||
|  |     data["is_response"] = is_response | ||||||
|  |     if _type == QuickReplyText._type: | ||||||
|  |         if q.get("title") is not None: | ||||||
|  |             data["title"] = q["title"] | ||||||
|  |         rtn = QuickReplyText(**data) | ||||||
|  |     elif _type == QuickReplyLocation._type: | ||||||
|  |         rtn = QuickReplyLocation(**data) | ||||||
|  |     elif _type == QuickReplyPhoneNumber._type: | ||||||
|  |         rtn = QuickReplyPhoneNumber(**data) | ||||||
|  |     elif _type == QuickReplyEmail._type: | ||||||
|  |         rtn = QuickReplyEmail(**data) | ||||||
|  |     return rtn | ||||||
							
								
								
									
										331
									
								
								fbchat/_state.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,331 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | import bs4 | ||||||
|  | import re | ||||||
|  | import requests | ||||||
|  | import random | ||||||
|  |  | ||||||
|  | from . import _graphql, _util, _exception | ||||||
|  |  | ||||||
|  | FB_DTSG_REGEX = re.compile(r'name="fb_dtsg" value="(.*?)"') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_user_id(session): | ||||||
|  |     # TODO: Optimize this `.get_dict()` call! | ||||||
|  |     rtn = session.cookies.get_dict().get("c_user") | ||||||
|  |     if rtn is None: | ||||||
|  |         raise _exception.FBchatException("Could not find user id") | ||||||
|  |     return str(rtn) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def find_input_fields(html): | ||||||
|  |     return bs4.BeautifulSoup(html, "html.parser", parse_only=bs4.SoupStrainer("input")) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def session_factory(user_agent=None): | ||||||
|  |     session = requests.session() | ||||||
|  |     session.headers["Referer"] = "https://www.facebook.com" | ||||||
|  |     # TODO: Deprecate setting the user agent manually | ||||||
|  |     session.headers["User-Agent"] = user_agent or random.choice(_util.USER_AGENTS) | ||||||
|  |     return session | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def client_id_factory(): | ||||||
|  |     return hex(int(random.random() * 2 ** 31))[2:] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def is_home(url): | ||||||
|  |     parts = _util.urlparse(url) | ||||||
|  |     # Check the urls `/home.php` and `/` | ||||||
|  |     return "home" in parts.path or "/" == parts.path | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _2fa_helper(session, code, r): | ||||||
|  |     soup = find_input_fields(r.text) | ||||||
|  |     data = dict() | ||||||
|  |  | ||||||
|  |     url = "https://m.facebook.com/login/checkpoint/" | ||||||
|  |  | ||||||
|  |     data["approvals_code"] = code | ||||||
|  |     data["fb_dtsg"] = soup.find("input", {"name": "fb_dtsg"})["value"] | ||||||
|  |     data["nh"] = soup.find("input", {"name": "nh"})["value"] | ||||||
|  |     data["submit[Submit Code]"] = "Submit Code" | ||||||
|  |     data["codes_submitted"] = 0 | ||||||
|  |     _util.log.info("Submitting 2FA code.") | ||||||
|  |  | ||||||
|  |     r = session.post(url, data=data) | ||||||
|  |  | ||||||
|  |     if is_home(r.url): | ||||||
|  |         return r | ||||||
|  |  | ||||||
|  |     del data["approvals_code"] | ||||||
|  |     del data["submit[Submit Code]"] | ||||||
|  |     del data["codes_submitted"] | ||||||
|  |  | ||||||
|  |     data["name_action_selected"] = "save_device" | ||||||
|  |     data["submit[Continue]"] = "Continue" | ||||||
|  |     _util.log.info("Saving browser.") | ||||||
|  |     # At this stage, we have dtsg, nh, name_action_selected, submit[Continue] | ||||||
|  |     r = session.post(url, data=data) | ||||||
|  |  | ||||||
|  |     if is_home(r.url): | ||||||
|  |         return r | ||||||
|  |  | ||||||
|  |     del data["name_action_selected"] | ||||||
|  |     _util.log.info("Starting Facebook checkup flow.") | ||||||
|  |     # At this stage, we have dtsg, nh, submit[Continue] | ||||||
|  |     r = session.post(url, data=data) | ||||||
|  |  | ||||||
|  |     if is_home(r.url): | ||||||
|  |         return r | ||||||
|  |  | ||||||
|  |     del data["submit[Continue]"] | ||||||
|  |     data["submit[This was me]"] = "This Was Me" | ||||||
|  |     _util.log.info("Verifying login attempt.") | ||||||
|  |     # At this stage, we have dtsg, nh, submit[This was me] | ||||||
|  |     r = session.post(url, data=data) | ||||||
|  |  | ||||||
|  |     if is_home(r.url): | ||||||
|  |         return r | ||||||
|  |  | ||||||
|  |     del data["submit[This was me]"] | ||||||
|  |     data["submit[Continue]"] = "Continue" | ||||||
|  |     data["name_action_selected"] = "save_device" | ||||||
|  |     _util.log.info("Saving device again.") | ||||||
|  |     # At this stage, we have dtsg, nh, submit[Continue], name_action_selected | ||||||
|  |     r = session.post(url, data=data) | ||||||
|  |     return r | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(slots=True)  # TODO i Python 3: Add kw_only=True | ||||||
|  | class State(object): | ||||||
|  |     """Stores and manages state required for most Facebook requests.""" | ||||||
|  |  | ||||||
|  |     user_id = attr.ib() | ||||||
|  |     _fb_dtsg = attr.ib() | ||||||
|  |     _revision = attr.ib() | ||||||
|  |     _session = attr.ib(factory=session_factory) | ||||||
|  |     _counter = attr.ib(0) | ||||||
|  |     _client_id = attr.ib(factory=client_id_factory) | ||||||
|  |     _logout_h = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def get_params(self): | ||||||
|  |         self._counter += 1  # TODO: Make this operation atomic / thread-safe | ||||||
|  |         return { | ||||||
|  |             "__a": 1, | ||||||
|  |             "__req": _util.str_base(self._counter, 36), | ||||||
|  |             "__rev": self._revision, | ||||||
|  |             "fb_dtsg": self._fb_dtsg, | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def login(cls, email, password, on_2fa_callback, user_agent=None): | ||||||
|  |         session = session_factory(user_agent=user_agent) | ||||||
|  |  | ||||||
|  |         soup = find_input_fields(session.get("https://m.facebook.com/").text) | ||||||
|  |         data = dict( | ||||||
|  |             (elem["name"], elem["value"]) | ||||||
|  |             for elem in soup | ||||||
|  |             if elem.has_attr("value") and elem.has_attr("name") | ||||||
|  |         ) | ||||||
|  |         data["email"] = email | ||||||
|  |         data["pass"] = password | ||||||
|  |         data["login"] = "Log In" | ||||||
|  |  | ||||||
|  |         r = session.post("https://m.facebook.com/login.php?login_attempt=1", data=data) | ||||||
|  |  | ||||||
|  |         # Usually, 'Checkpoint' will refer to 2FA | ||||||
|  |         if "checkpoint" in r.url and ('id="approvals_code"' in r.text.lower()): | ||||||
|  |             code = on_2fa_callback() | ||||||
|  |             r = _2fa_helper(session, code, r) | ||||||
|  |  | ||||||
|  |         # Sometimes Facebook tries to show the user a "Save Device" dialog | ||||||
|  |         if "save-device" in r.url: | ||||||
|  |             r = session.get("https://m.facebook.com/login/save-device/cancel/") | ||||||
|  |  | ||||||
|  |         if is_home(r.url): | ||||||
|  |             return cls.from_session(session=session) | ||||||
|  |         else: | ||||||
|  |             raise _exception.FBchatUserError( | ||||||
|  |                 "Login failed. Check email/password. " | ||||||
|  |                 "(Failed on url: {})".format(r.url) | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |     def is_logged_in(self): | ||||||
|  |         # Send a request to the login url, to see if we're directed to the home page | ||||||
|  |         url = "https://m.facebook.com/login.php?login_attempt=1" | ||||||
|  |         r = self._session.get(url, allow_redirects=False) | ||||||
|  |         return "Location" in r.headers and is_home(r.headers["Location"]) | ||||||
|  |  | ||||||
|  |     def logout(self): | ||||||
|  |         logout_h = self._logout_h | ||||||
|  |         if not logout_h: | ||||||
|  |             url = _util.prefix_url("/bluebar/modern_settings_menu/") | ||||||
|  |             h_r = self._session.post(url, data={"pmid": "4"}) | ||||||
|  |             logout_h = re.search(r'name=\\"h\\" value=\\"(.*?)\\"', h_r.text).group(1) | ||||||
|  |  | ||||||
|  |         url = _util.prefix_url("/logout.php") | ||||||
|  |         return self._session.get(url, params={"ref": "mb", "h": logout_h}).ok | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def from_session(cls, session): | ||||||
|  |         # TODO: Automatically set user_id when the cookie changes in the session | ||||||
|  |         user_id = get_user_id(session) | ||||||
|  |  | ||||||
|  |         r = session.get(_util.prefix_url("/")) | ||||||
|  |  | ||||||
|  |         soup = find_input_fields(r.text) | ||||||
|  |  | ||||||
|  |         fb_dtsg_element = soup.find("input", {"name": "fb_dtsg"}) | ||||||
|  |         if fb_dtsg_element: | ||||||
|  |             fb_dtsg = fb_dtsg_element["value"] | ||||||
|  |         else: | ||||||
|  |             # Fall back to searching with a regex | ||||||
|  |             fb_dtsg = FB_DTSG_REGEX.search(r.text).group(1) | ||||||
|  |  | ||||||
|  |         revision = int(r.text.split('"client_revision":', 1)[1].split(",", 1)[0]) | ||||||
|  |  | ||||||
|  |         logout_h_element = soup.find("input", {"name": "h"}) | ||||||
|  |         logout_h = logout_h_element["value"] if logout_h_element else None | ||||||
|  |  | ||||||
|  |         return cls( | ||||||
|  |             user_id=user_id, | ||||||
|  |             fb_dtsg=fb_dtsg, | ||||||
|  |             revision=revision, | ||||||
|  |             session=session, | ||||||
|  |             logout_h=logout_h, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def get_cookies(self): | ||||||
|  |         return self._session.cookies.get_dict() | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def from_cookies(cls, cookies, user_agent=None): | ||||||
|  |         session = session_factory(user_agent=user_agent) | ||||||
|  |         session.cookies = requests.cookies.merge_cookies(session.cookies, cookies) | ||||||
|  |         return cls.from_session(session=session) | ||||||
|  |  | ||||||
|  |     def _do_refresh(self): | ||||||
|  |         # TODO: Raise the error instead, and make the user do the refresh manually | ||||||
|  |         # It may be a bad idea to do this in an exception handler, if you have a better method, please suggest it! | ||||||
|  |         _util.log.warning("Refreshing state and resending request") | ||||||
|  |         new = State.from_session(session=self._session) | ||||||
|  |         self.user_id = new.user_id | ||||||
|  |         self._fb_dtsg = new._fb_dtsg | ||||||
|  |         self._revision = new._revision | ||||||
|  |         self._counter = new._counter | ||||||
|  |         self._logout_h = new._logout_h or self._logout_h | ||||||
|  |  | ||||||
|  |     def _get(self, url, params, error_retries=3): | ||||||
|  |         params.update(self.get_params()) | ||||||
|  |         r = self._session.get(_util.prefix_url(url), params=params) | ||||||
|  |         content = _util.check_request(r) | ||||||
|  |         j = _util.to_json(content) | ||||||
|  |         try: | ||||||
|  |             _util.handle_payload_error(j) | ||||||
|  |         except _exception.FBchatPleaseRefresh: | ||||||
|  |             if error_retries > 0: | ||||||
|  |                 self._do_refresh() | ||||||
|  |                 return self._get(url, params, error_retries=error_retries - 1) | ||||||
|  |             raise | ||||||
|  |         return j | ||||||
|  |  | ||||||
|  |     def _post(self, url, data, files=None, as_graphql=False, error_retries=3): | ||||||
|  |         data.update(self.get_params()) | ||||||
|  |         r = self._session.post(_util.prefix_url(url), data=data, files=files) | ||||||
|  |         content = _util.check_request(r) | ||||||
|  |         try: | ||||||
|  |             if as_graphql: | ||||||
|  |                 return _graphql.response_to_json(content) | ||||||
|  |             else: | ||||||
|  |                 j = _util.to_json(content) | ||||||
|  |                 # TODO: Remove this, and move it to _payload_post instead | ||||||
|  |                 # We can't yet, since errors raised in here need to be caught below | ||||||
|  |                 _util.handle_payload_error(j) | ||||||
|  |                 return j | ||||||
|  |         except _exception.FBchatPleaseRefresh: | ||||||
|  |             if error_retries > 0: | ||||||
|  |                 self._do_refresh() | ||||||
|  |                 return self._post( | ||||||
|  |                     url, | ||||||
|  |                     data, | ||||||
|  |                     files=files, | ||||||
|  |                     as_graphql=as_graphql, | ||||||
|  |                     error_retries=error_retries - 1, | ||||||
|  |                 ) | ||||||
|  |             raise | ||||||
|  |  | ||||||
|  |     def _payload_post(self, url, data, files=None): | ||||||
|  |         j = self._post(url, data, files=files) | ||||||
|  |         try: | ||||||
|  |             return j["payload"] | ||||||
|  |         except (KeyError, TypeError): | ||||||
|  |             raise _exception.FBchatException("Missing payload: {}".format(j)) | ||||||
|  |  | ||||||
|  |     def _graphql_requests(self, *queries): | ||||||
|  |         data = { | ||||||
|  |             "method": "GET", | ||||||
|  |             "response_format": "json", | ||||||
|  |             "queries": _graphql.queries_to_json(*queries), | ||||||
|  |         } | ||||||
|  |         return self._post("/api/graphqlbatch/", data, as_graphql=True) | ||||||
|  |  | ||||||
|  |     def _upload(self, files, voice_clip=False): | ||||||
|  |         """Upload files to Facebook. | ||||||
|  |  | ||||||
|  |         `files` should be a list of files that requests can upload, see | ||||||
|  |         `requests.request <https://docs.python-requests.org/en/master/api/#requests.request>`_. | ||||||
|  |  | ||||||
|  |         Return a list of tuples with a file's ID and mimetype. | ||||||
|  |         """ | ||||||
|  |         file_dict = {"upload_{}".format(i): f for i, f in enumerate(files)} | ||||||
|  |  | ||||||
|  |         data = {"voice_clip": voice_clip} | ||||||
|  |  | ||||||
|  |         j = self._payload_post( | ||||||
|  |             "https://upload.facebook.com/ajax/mercury/upload.php", data, files=file_dict | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         if len(j["metadata"]) != len(files): | ||||||
|  |             raise _exception.FBchatException( | ||||||
|  |                 "Some files could not be uploaded: {}, {}".format(j, files) | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |         return [ | ||||||
|  |             (data[_util.mimetype_to_key(data["filetype"])], data["filetype"]) | ||||||
|  |             for data in j["metadata"] | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     def _do_send_request(self, data): | ||||||
|  |         offline_threading_id = _util.generateOfflineThreadingID() | ||||||
|  |         data["client"] = "mercury" | ||||||
|  |         data["author"] = "fbid:{}".format(self.user_id) | ||||||
|  |         data["timestamp"] = _util.now() | ||||||
|  |         data["source"] = "source:chat:web" | ||||||
|  |         data["offline_threading_id"] = offline_threading_id | ||||||
|  |         data["message_id"] = offline_threading_id | ||||||
|  |         data["threading_id"] = _util.generateMessageID(self._client_id) | ||||||
|  |         data["ephemeral_ttl_mode:"] = "0" | ||||||
|  |         j = self._post("/messaging/send/", data) | ||||||
|  |  | ||||||
|  |         # update JS token if received in response | ||||||
|  |         fb_dtsg = _util.get_jsmods_require(j, 2) | ||||||
|  |         if fb_dtsg is not None: | ||||||
|  |             self._fb_dtsg = fb_dtsg | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             message_ids = [ | ||||||
|  |                 (action["message_id"], action["thread_fbid"]) | ||||||
|  |                 for action in j["payload"]["actions"] | ||||||
|  |                 if "message_id" in action | ||||||
|  |             ] | ||||||
|  |             if len(message_ids) != 1: | ||||||
|  |                 log.warning("Got multiple message ids' back: {}".format(message_ids)) | ||||||
|  |             return message_ids[0] | ||||||
|  |         except (KeyError, IndexError, TypeError) as e: | ||||||
|  |             raise _exception.FBchatException( | ||||||
|  |                 "Error when sending message: " | ||||||
|  |                 "No message IDs could be found: {}".format(j) | ||||||
|  |             ) | ||||||
							
								
								
									
										60
									
								
								fbchat/_sticker.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,60 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | from ._attachment import Attachment | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class Sticker(Attachment): | ||||||
|  |     """Represents a Facebook sticker that has been sent to a thread as an attachment.""" | ||||||
|  |  | ||||||
|  |     #: The sticker-pack's ID | ||||||
|  |     pack = attr.ib(None) | ||||||
|  |     #: Whether the sticker is animated | ||||||
|  |     is_animated = attr.ib(False) | ||||||
|  |  | ||||||
|  |     # If the sticker is animated, the following should be present | ||||||
|  |     #: URL to a medium spritemap | ||||||
|  |     medium_sprite_image = attr.ib(None) | ||||||
|  |     #: URL to a large spritemap | ||||||
|  |     large_sprite_image = attr.ib(None) | ||||||
|  |     #: The amount of frames present in the spritemap pr. row | ||||||
|  |     frames_per_row = attr.ib(None) | ||||||
|  |     #: The amount of frames present in the spritemap pr. column | ||||||
|  |     frames_per_col = attr.ib(None) | ||||||
|  |     #: The frame rate the spritemap is intended to be played in | ||||||
|  |     frame_rate = attr.ib(None) | ||||||
|  |  | ||||||
|  |     #: URL to the sticker's image | ||||||
|  |     url = attr.ib(None) | ||||||
|  |     #: Width of the sticker | ||||||
|  |     width = attr.ib(None) | ||||||
|  |     #: Height of the sticker | ||||||
|  |     height = attr.ib(None) | ||||||
|  |     #: The sticker's label/name | ||||||
|  |     label = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def __init__(self, uid=None): | ||||||
|  |         super(Sticker, self).__init__(uid=uid) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         if not data: | ||||||
|  |             return None | ||||||
|  |         self = cls(uid=data["id"]) | ||||||
|  |         if data.get("pack"): | ||||||
|  |             self.pack = data["pack"].get("id") | ||||||
|  |         if data.get("sprite_image"): | ||||||
|  |             self.is_animated = True | ||||||
|  |             self.medium_sprite_image = data["sprite_image"].get("uri") | ||||||
|  |             self.large_sprite_image = data["sprite_image_2x"].get("uri") | ||||||
|  |             self.frames_per_row = data.get("frames_per_row") | ||||||
|  |             self.frames_per_col = data.get("frames_per_column") | ||||||
|  |             self.frame_rate = data.get("frame_rate") | ||||||
|  |         self.url = data.get("url") | ||||||
|  |         self.width = data.get("width") | ||||||
|  |         self.height = data.get("height") | ||||||
|  |         if data.get("label"): | ||||||
|  |             self.label = data["label"] | ||||||
|  |         return self | ||||||
							
								
								
									
										147
									
								
								fbchat/_thread.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,147 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | from ._core import Enum | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ThreadType(Enum): | ||||||
|  |     """Used to specify what type of Facebook thread is being used. | ||||||
|  |  | ||||||
|  |     See :ref:`intro_threads` for more info. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     USER = 1 | ||||||
|  |     GROUP = 2 | ||||||
|  |     ROOM = 2 | ||||||
|  |     PAGE = 3 | ||||||
|  |  | ||||||
|  |     def _to_class(self): | ||||||
|  |         """Convert this enum value to the corresponding class.""" | ||||||
|  |         from . import _user, _group, _page | ||||||
|  |  | ||||||
|  |         return { | ||||||
|  |             ThreadType.USER: _user.User, | ||||||
|  |             ThreadType.GROUP: _group.Group, | ||||||
|  |             ThreadType.ROOM: _group.Room, | ||||||
|  |             ThreadType.PAGE: _page.Page, | ||||||
|  |         }[self] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ThreadLocation(Enum): | ||||||
|  |     """Used to specify where a thread is located (inbox, pending, archived, other).""" | ||||||
|  |  | ||||||
|  |     INBOX = "INBOX" | ||||||
|  |     PENDING = "PENDING" | ||||||
|  |     ARCHIVED = "ARCHIVED" | ||||||
|  |     OTHER = "OTHER" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ThreadColor(Enum): | ||||||
|  |     """Used to specify a thread colors.""" | ||||||
|  |  | ||||||
|  |     MESSENGER_BLUE = "#0084ff" | ||||||
|  |     VIKING = "#44bec7" | ||||||
|  |     GOLDEN_POPPY = "#ffc300" | ||||||
|  |     RADICAL_RED = "#fa3c4c" | ||||||
|  |     SHOCKING = "#d696bb" | ||||||
|  |     PICTON_BLUE = "#6699cc" | ||||||
|  |     FREE_SPEECH_GREEN = "#13cf13" | ||||||
|  |     PUMPKIN = "#ff7e29" | ||||||
|  |     LIGHT_CORAL = "#e68585" | ||||||
|  |     MEDIUM_SLATE_BLUE = "#7646ff" | ||||||
|  |     DEEP_SKY_BLUE = "#20cef5" | ||||||
|  |     FERN = "#67b868" | ||||||
|  |     CAMEO = "#d4a88c" | ||||||
|  |     BRILLIANT_ROSE = "#ff5ca1" | ||||||
|  |     BILOBA_FLOWER = "#a695c7" | ||||||
|  |     TICKLE_ME_PINK = "#ff7ca8" | ||||||
|  |     MALACHITE = "#1adb5b" | ||||||
|  |     RUBY = "#f01d6a" | ||||||
|  |     DARK_TANGERINE = "#ff9c19" | ||||||
|  |     BRIGHT_TURQUOISE = "#0edcde" | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, color): | ||||||
|  |         if color is None: | ||||||
|  |             return None | ||||||
|  |         if not color: | ||||||
|  |             return cls.MESSENGER_BLUE | ||||||
|  |         color = color[2:]  # Strip the alpha value | ||||||
|  |         value = "#{}".format(color.lower()) | ||||||
|  |         return cls._extend_if_invalid(value) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class Thread(object): | ||||||
|  |     """Represents a Facebook thread.""" | ||||||
|  |  | ||||||
|  |     #: The unique identifier of the thread. Can be used a ``thread_id``. See :ref:`intro_threads` for more info | ||||||
|  |     uid = attr.ib(converter=str) | ||||||
|  |     #: Specifies the type of thread. Can be used a ``thread_type``. See :ref:`intro_threads` for more info | ||||||
|  |     type = attr.ib() | ||||||
|  |     #: A URL to the thread's picture | ||||||
|  |     photo = attr.ib(None) | ||||||
|  |     #: The name of the thread | ||||||
|  |     name = attr.ib(None) | ||||||
|  |     #: Timestamp of last message | ||||||
|  |     last_message_timestamp = attr.ib(None) | ||||||
|  |     #: Number of messages in the thread | ||||||
|  |     message_count = attr.ib(None) | ||||||
|  |     #: Set :class:`Plan` | ||||||
|  |     plan = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         _type, | ||||||
|  |         uid, | ||||||
|  |         photo=None, | ||||||
|  |         name=None, | ||||||
|  |         last_message_timestamp=None, | ||||||
|  |         message_count=None, | ||||||
|  |         plan=None, | ||||||
|  |     ): | ||||||
|  |         self.uid = str(uid) | ||||||
|  |         self.type = _type | ||||||
|  |         self.photo = photo | ||||||
|  |         self.name = name | ||||||
|  |         self.last_message_timestamp = last_message_timestamp | ||||||
|  |         self.message_count = message_count | ||||||
|  |         self.plan = plan | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def _parse_customization_info(data): | ||||||
|  |         if data is None or data.get("customization_info") is None: | ||||||
|  |             return {} | ||||||
|  |         info = data["customization_info"] | ||||||
|  |  | ||||||
|  |         rtn = { | ||||||
|  |             "emoji": info.get("emoji"), | ||||||
|  |             "color": ThreadColor._from_graphql(info.get("outgoing_bubble_color")), | ||||||
|  |         } | ||||||
|  |         if ( | ||||||
|  |             data.get("thread_type") == "GROUP" | ||||||
|  |             or data.get("is_group_thread") | ||||||
|  |             or data.get("thread_key", {}).get("thread_fbid") | ||||||
|  |         ): | ||||||
|  |             rtn["nicknames"] = {} | ||||||
|  |             for k in info.get("participant_customizations", []): | ||||||
|  |                 rtn["nicknames"][k["participant_id"]] = k.get("nickname") | ||||||
|  |         elif info.get("participant_customizations"): | ||||||
|  |             uid = data.get("thread_key", {}).get("other_user_id") or data.get("id") | ||||||
|  |             pc = info["participant_customizations"] | ||||||
|  |             if len(pc) > 0: | ||||||
|  |                 if pc[0].get("participant_id") == uid: | ||||||
|  |                     rtn["nickname"] = pc[0].get("nickname") | ||||||
|  |                 else: | ||||||
|  |                     rtn["own_nickname"] = pc[0].get("nickname") | ||||||
|  |             if len(pc) > 1: | ||||||
|  |                 if pc[1].get("participant_id") == uid: | ||||||
|  |                     rtn["nickname"] = pc[1].get("nickname") | ||||||
|  |                 else: | ||||||
|  |                     rtn["own_nickname"] = pc[1].get("nickname") | ||||||
|  |         return rtn | ||||||
|  |  | ||||||
|  |     def _to_send_data(self): | ||||||
|  |         # TODO: Only implement this in subclasses | ||||||
|  |         return {"other_user_fbid": self.uid} | ||||||
							
								
								
									
										197
									
								
								fbchat/_user.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,197 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import attr | ||||||
|  | from ._core import Enum | ||||||
|  | from . import _plan | ||||||
|  | from ._thread import ThreadType, Thread | ||||||
|  |  | ||||||
|  |  | ||||||
|  | GENDERS = { | ||||||
|  |     # For standard requests | ||||||
|  |     0: "unknown", | ||||||
|  |     1: "female_singular", | ||||||
|  |     2: "male_singular", | ||||||
|  |     3: "female_singular_guess", | ||||||
|  |     4: "male_singular_guess", | ||||||
|  |     5: "mixed", | ||||||
|  |     6: "neuter_singular", | ||||||
|  |     7: "unknown_singular", | ||||||
|  |     8: "female_plural", | ||||||
|  |     9: "male_plural", | ||||||
|  |     10: "neuter_plural", | ||||||
|  |     11: "unknown_plural", | ||||||
|  |     # For graphql requests | ||||||
|  |     "UNKNOWN": "unknown", | ||||||
|  |     "FEMALE": "female_singular", | ||||||
|  |     "MALE": "male_singular", | ||||||
|  |     # '': 'female_singular_guess', | ||||||
|  |     # '': 'male_singular_guess', | ||||||
|  |     # '': 'mixed', | ||||||
|  |     "NEUTER": "neuter_singular", | ||||||
|  |     # '': 'unknown_singular', | ||||||
|  |     # '': 'female_plural', | ||||||
|  |     # '': 'male_plural', | ||||||
|  |     # '': 'neuter_plural', | ||||||
|  |     # '': 'unknown_plural', | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TypingStatus(Enum): | ||||||
|  |     """Used to specify whether the user is typing or has stopped typing.""" | ||||||
|  |  | ||||||
|  |     STOPPED = 0 | ||||||
|  |     TYPING = 1 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False, init=False) | ||||||
|  | class User(Thread): | ||||||
|  |     """Represents a Facebook user. Inherits `Thread`.""" | ||||||
|  |  | ||||||
|  |     #: The profile URL | ||||||
|  |     url = attr.ib(None) | ||||||
|  |     #: The users first name | ||||||
|  |     first_name = attr.ib(None) | ||||||
|  |     #: The users last name | ||||||
|  |     last_name = attr.ib(None) | ||||||
|  |     #: Whether the user and the client are friends | ||||||
|  |     is_friend = attr.ib(None) | ||||||
|  |     #: The user's gender | ||||||
|  |     gender = attr.ib(None) | ||||||
|  |     #: From 0 to 1. How close the client is to the user | ||||||
|  |     affinity = attr.ib(None) | ||||||
|  |     #: The user's nickname | ||||||
|  |     nickname = attr.ib(None) | ||||||
|  |     #: The clients nickname, as seen by the user | ||||||
|  |     own_nickname = attr.ib(None) | ||||||
|  |     #: A :class:`ThreadColor`. The message color | ||||||
|  |     color = attr.ib(None) | ||||||
|  |     #: The default emoji | ||||||
|  |     emoji = attr.ib(None) | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         uid, | ||||||
|  |         url=None, | ||||||
|  |         first_name=None, | ||||||
|  |         last_name=None, | ||||||
|  |         is_friend=None, | ||||||
|  |         gender=None, | ||||||
|  |         affinity=None, | ||||||
|  |         nickname=None, | ||||||
|  |         own_nickname=None, | ||||||
|  |         color=None, | ||||||
|  |         emoji=None, | ||||||
|  |         **kwargs | ||||||
|  |     ): | ||||||
|  |         super(User, self).__init__(ThreadType.USER, uid, **kwargs) | ||||||
|  |         self.url = url | ||||||
|  |         self.first_name = first_name | ||||||
|  |         self.last_name = last_name | ||||||
|  |         self.is_friend = is_friend | ||||||
|  |         self.gender = gender | ||||||
|  |         self.affinity = affinity | ||||||
|  |         self.nickname = nickname | ||||||
|  |         self.own_nickname = own_nickname | ||||||
|  |         self.color = color | ||||||
|  |         self.emoji = emoji | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_graphql(cls, data): | ||||||
|  |         if data.get("profile_picture") is None: | ||||||
|  |             data["profile_picture"] = {} | ||||||
|  |         c_info = cls._parse_customization_info(data) | ||||||
|  |         plan = None | ||||||
|  |         if data.get("event_reminders") and data["event_reminders"].get("nodes"): | ||||||
|  |             plan = _plan.Plan._from_graphql(data["event_reminders"]["nodes"][0]) | ||||||
|  |  | ||||||
|  |         return cls( | ||||||
|  |             data["id"], | ||||||
|  |             url=data.get("url"), | ||||||
|  |             first_name=data.get("first_name"), | ||||||
|  |             last_name=data.get("last_name"), | ||||||
|  |             is_friend=data.get("is_viewer_friend"), | ||||||
|  |             gender=GENDERS.get(data.get("gender")), | ||||||
|  |             affinity=data.get("affinity"), | ||||||
|  |             nickname=c_info.get("nickname"), | ||||||
|  |             color=c_info.get("color"), | ||||||
|  |             emoji=c_info.get("emoji"), | ||||||
|  |             own_nickname=c_info.get("own_nickname"), | ||||||
|  |             photo=data["profile_picture"].get("uri"), | ||||||
|  |             name=data.get("name"), | ||||||
|  |             message_count=data.get("messages_count"), | ||||||
|  |             plan=plan, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_thread_fetch(cls, data): | ||||||
|  |         if data.get("big_image_src") is None: | ||||||
|  |             data["big_image_src"] = {} | ||||||
|  |         c_info = cls._parse_customization_info(data) | ||||||
|  |         participants = [ | ||||||
|  |             node["messaging_actor"] for node in data["all_participants"]["nodes"] | ||||||
|  |         ] | ||||||
|  |         user = next( | ||||||
|  |             p for p in participants if p["id"] == data["thread_key"]["other_user_id"] | ||||||
|  |         ) | ||||||
|  |         last_message_timestamp = None | ||||||
|  |         if "last_message" in data: | ||||||
|  |             last_message_timestamp = data["last_message"]["nodes"][0][ | ||||||
|  |                 "timestamp_precise" | ||||||
|  |             ] | ||||||
|  |  | ||||||
|  |         first_name = user.get("short_name") | ||||||
|  |         if first_name is None: | ||||||
|  |             last_name = None | ||||||
|  |         else: | ||||||
|  |             last_name = user.get("name").split(first_name, 1).pop().strip() | ||||||
|  |  | ||||||
|  |         plan = None | ||||||
|  |         if data.get("event_reminders") and data["event_reminders"].get("nodes"): | ||||||
|  |             plan = _plan.Plan._from_graphql(data["event_reminders"]["nodes"][0]) | ||||||
|  |  | ||||||
|  |         return cls( | ||||||
|  |             user["id"], | ||||||
|  |             url=user.get("url"), | ||||||
|  |             name=user.get("name"), | ||||||
|  |             first_name=first_name, | ||||||
|  |             last_name=last_name, | ||||||
|  |             is_friend=user.get("is_viewer_friend"), | ||||||
|  |             gender=GENDERS.get(user.get("gender")), | ||||||
|  |             affinity=user.get("affinity"), | ||||||
|  |             nickname=c_info.get("nickname"), | ||||||
|  |             color=c_info.get("color"), | ||||||
|  |             emoji=c_info.get("emoji"), | ||||||
|  |             own_nickname=c_info.get("own_nickname"), | ||||||
|  |             photo=user["big_image_src"].get("uri"), | ||||||
|  |             message_count=data.get("messages_count"), | ||||||
|  |             last_message_timestamp=last_message_timestamp, | ||||||
|  |             plan=plan, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_all_fetch(cls, data): | ||||||
|  |         return cls( | ||||||
|  |             data["id"], | ||||||
|  |             first_name=data.get("firstName"), | ||||||
|  |             url=data.get("uri"), | ||||||
|  |             photo=data.get("thumbSrc"), | ||||||
|  |             name=data.get("name"), | ||||||
|  |             is_friend=data.get("is_friend"), | ||||||
|  |             gender=GENDERS.get(data.get("gender")), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @attr.s(cmp=False) | ||||||
|  | class ActiveStatus(object): | ||||||
|  |     #: Whether the user is active now | ||||||
|  |     active = attr.ib(None) | ||||||
|  |     #: Timestamp when the user was last active | ||||||
|  |     last_active = attr.ib(None) | ||||||
|  |     #: Whether the user is playing Messenger game now | ||||||
|  |     in_game = attr.ib(None) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _from_orca_presence(cls, data): | ||||||
|  |         # TODO: Handle `c` and `vc` keys (Probably some binary data) | ||||||
|  |         return cls(active=data["p"] in [2, 3], last_active=data.get("l"), in_game=None) | ||||||
							
								
								
									
										270
									
								
								fbchat/_util.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,270 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  | import re | ||||||
|  | import json | ||||||
|  | from time import time | ||||||
|  | from random import random | ||||||
|  | from contextlib import contextmanager | ||||||
|  | from mimetypes import guess_type | ||||||
|  | from os.path import basename | ||||||
|  | import warnings | ||||||
|  | import logging | ||||||
|  | import requests | ||||||
|  | from ._exception import ( | ||||||
|  |     FBchatException, | ||||||
|  |     FBchatFacebookError, | ||||||
|  |     FBchatInvalidParameters, | ||||||
|  |     FBchatNotLoggedIn, | ||||||
|  |     FBchatPleaseRefresh, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     from urllib.parse import urlencode, parse_qs, urlparse | ||||||
|  |  | ||||||
|  |     basestring = (str, bytes) | ||||||
|  | except ImportError: | ||||||
|  |     from urllib import urlencode | ||||||
|  |     from urlparse import parse_qs, urlparse | ||||||
|  |  | ||||||
|  |     basestring = basestring | ||||||
|  |  | ||||||
|  | # Python 2's `input` executes the input, whereas `raw_input` just returns the input | ||||||
|  | try: | ||||||
|  |     input = raw_input | ||||||
|  | except NameError: | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  | # Log settings | ||||||
|  | log = logging.getLogger("client") | ||||||
|  | log.setLevel(logging.DEBUG) | ||||||
|  | # Creates the console handler | ||||||
|  | handler = logging.StreamHandler() | ||||||
|  | log.addHandler(handler) | ||||||
|  |  | ||||||
|  | #: Default list of user agents | ||||||
|  | USER_AGENTS = [ | ||||||
|  |     "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/42.0.2311.90 Safari/537.36", | ||||||
|  |     "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_3) AppleWebKit/601.1.10 (KHTML, like Gecko) Version/8.0.5 Safari/601.1.10", | ||||||
|  |     "Mozilla/5.0 (Windows NT 6.3; WOW64; ; NCT50_AAP285C84A1328) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/42.0.2311.90 Safari/537.36", | ||||||
|  |     "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.1 (KHTML, like Gecko) Chrome/22.0.1207.1 Safari/537.1", | ||||||
|  |     "Mozilla/5.0 (X11; CrOS i686 2268.111.0) AppleWebKit/536.11 (KHTML, like Gecko) Chrome/20.0.1132.57 Safari/536.11", | ||||||
|  |     "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.6 (KHTML, like Gecko) Chrome/20.0.1092.0 Safari/536.6", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def now(): | ||||||
|  |     return int(time() * 1000) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def json_minimal(data): | ||||||
|  |     """Get JSON data in minimal form.""" | ||||||
|  |     return json.dumps(data, separators=(",", ":")) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def strip_json_cruft(text): | ||||||
|  |     """Removes `for(;;);` (and other cruft) that preceeds JSON responses.""" | ||||||
|  |     try: | ||||||
|  |         return text[text.index("{") :] | ||||||
|  |     except ValueError: | ||||||
|  |         raise FBchatException("No JSON object found: {!r}".format(text)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_cookie_header(session, url): | ||||||
|  |     """Extract a cookie header from a requests session.""" | ||||||
|  |     # The cookies are extracted this way to make sure they're escaped correctly | ||||||
|  |     return requests.cookies.get_cookie_header( | ||||||
|  |         session.cookies, requests.Request("GET", url), | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_decoded_r(r): | ||||||
|  |     return get_decoded(r._content) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_decoded(content): | ||||||
|  |     return content.decode("utf-8") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def parse_json(content): | ||||||
|  |     try: | ||||||
|  |         return json.loads(content) | ||||||
|  |     except ValueError: | ||||||
|  |         raise FBchatFacebookError("Error while parsing JSON: {!r}".format(content)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def digitToChar(digit): | ||||||
|  |     if digit < 10: | ||||||
|  |         return str(digit) | ||||||
|  |     return chr(ord("a") + digit - 10) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def str_base(number, base): | ||||||
|  |     if number < 0: | ||||||
|  |         return "-" + str_base(-number, base) | ||||||
|  |     (d, m) = divmod(number, base) | ||||||
|  |     if d > 0: | ||||||
|  |         return str_base(d, base) + digitToChar(m) | ||||||
|  |     return digitToChar(m) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def generateMessageID(client_id=None): | ||||||
|  |     k = now() | ||||||
|  |     l = int(random() * 4294967295) | ||||||
|  |     return "<{}:{}-{}@mail.projektitan.com>".format(k, l, client_id) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def getSignatureID(): | ||||||
|  |     return hex(int(random() * 2147483648)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def generateOfflineThreadingID(): | ||||||
|  |     ret = now() | ||||||
|  |     value = int(random() * 4294967295) | ||||||
|  |     string = ("0000000000000000000000" + format(value, "b"))[-22:] | ||||||
|  |     msgs = format(ret, "b") + string | ||||||
|  |     return str(int(msgs, 2)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def handle_payload_error(j): | ||||||
|  |     if "error" not in j: | ||||||
|  |         return | ||||||
|  |     error = j["error"] | ||||||
|  |     if j["error"] == 1357001: | ||||||
|  |         error_cls = FBchatNotLoggedIn | ||||||
|  |     elif j["error"] == 1357004: | ||||||
|  |         error_cls = FBchatPleaseRefresh | ||||||
|  |     elif j["error"] in (1357031, 1545010, 1545003): | ||||||
|  |         error_cls = FBchatInvalidParameters | ||||||
|  |     else: | ||||||
|  |         error_cls = FBchatFacebookError | ||||||
|  |     # TODO: Use j["errorSummary"] | ||||||
|  |     # "errorDescription" is in the users own language! | ||||||
|  |     raise error_cls( | ||||||
|  |         "Error #{} when sending request: {}".format(error, j["errorDescription"]), | ||||||
|  |         fb_error_code=error, | ||||||
|  |         fb_error_message=j["errorDescription"], | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def handle_graphql_errors(j): | ||||||
|  |     errors = [] | ||||||
|  |     if j.get("error"): | ||||||
|  |         errors = [j["error"]] | ||||||
|  |     if "errors" in j: | ||||||
|  |         errors = j["errors"] | ||||||
|  |     if errors: | ||||||
|  |         error = errors[0]  # TODO: Handle multiple errors | ||||||
|  |         # TODO: Use `summary`, `severity` and `description` | ||||||
|  |         raise FBchatFacebookError( | ||||||
|  |             "GraphQL error #{}: {} / {!r}".format( | ||||||
|  |                 error.get("code"), error.get("message"), error.get("debug_info") | ||||||
|  |             ), | ||||||
|  |             fb_error_code=error.get("code"), | ||||||
|  |             fb_error_message=error.get("message"), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def check_request(r): | ||||||
|  |     check_http_code(r.status_code) | ||||||
|  |     content = get_decoded_r(r) | ||||||
|  |     check_content(content) | ||||||
|  |     return content | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def check_http_code(code): | ||||||
|  |     msg = "Error when sending request: Got {} response.".format(code) | ||||||
|  |     if code == 404: | ||||||
|  |         raise FBchatFacebookError( | ||||||
|  |             msg + " This is either because you specified an invalid URL, or because" | ||||||
|  |             " you provided an invalid id (Facebook usually requires integer ids).", | ||||||
|  |             request_status_code=code, | ||||||
|  |         ) | ||||||
|  |     if 400 <= code < 600: | ||||||
|  |         raise FBchatFacebookError(msg, request_status_code=code) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def check_content(content, as_json=True): | ||||||
|  |     if content is None or len(content) == 0: | ||||||
|  |         raise FBchatFacebookError("Error when sending request: Got empty response") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def to_json(content): | ||||||
|  |     content = strip_json_cruft(content) | ||||||
|  |     j = parse_json(content) | ||||||
|  |     log.debug(j) | ||||||
|  |     return j | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_jsmods_require(j, index): | ||||||
|  |     if j.get("jsmods") and j["jsmods"].get("require"): | ||||||
|  |         try: | ||||||
|  |             return j["jsmods"]["require"][0][index][0] | ||||||
|  |         except (KeyError, IndexError) as e: | ||||||
|  |             log.warning( | ||||||
|  |                 "Error when getting jsmods_require: " | ||||||
|  |                 "{}. Facebook might have changed protocol".format(j) | ||||||
|  |             ) | ||||||
|  |     return None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def require_list(list_): | ||||||
|  |     if isinstance(list_, list): | ||||||
|  |         return set(list_) | ||||||
|  |     else: | ||||||
|  |         return set([list_]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def mimetype_to_key(mimetype): | ||||||
|  |     if not mimetype: | ||||||
|  |         return "file_id" | ||||||
|  |     if mimetype == "image/gif": | ||||||
|  |         return "gif_id" | ||||||
|  |     x = mimetype.split("/") | ||||||
|  |     if x[0] in ["video", "image", "audio"]: | ||||||
|  |         return "%s_id" % x[0] | ||||||
|  |     return "file_id" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_files_from_urls(file_urls): | ||||||
|  |     files = [] | ||||||
|  |     for file_url in file_urls: | ||||||
|  |         r = requests.get(file_url) | ||||||
|  |         # We could possibly use r.headers.get('Content-Disposition'), see | ||||||
|  |         # https://stackoverflow.com/a/37060758 | ||||||
|  |         file_name = basename(file_url).split("?")[0].split("#")[0] | ||||||
|  |         files.append( | ||||||
|  |             ( | ||||||
|  |                 file_name, | ||||||
|  |                 r.content, | ||||||
|  |                 r.headers.get("Content-Type") or guess_type(file_name)[0], | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |     return files | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @contextmanager | ||||||
|  | def get_files_from_paths(filenames): | ||||||
|  |     files = [] | ||||||
|  |     for filename in filenames: | ||||||
|  |         files.append( | ||||||
|  |             (basename(filename), open(filename, "rb"), guess_type(filename)[0]) | ||||||
|  |         ) | ||||||
|  |     yield files | ||||||
|  |     for fn, fp, ft in files: | ||||||
|  |         fp.close() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_url_parameters(url, *args): | ||||||
|  |     params = parse_qs(urlparse(url).query) | ||||||
|  |     return [params[arg][0] for arg in args if params.get(arg)] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_url_parameter(url, param): | ||||||
|  |     return get_url_parameters(url, param)[0] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def prefix_url(url): | ||||||
|  |     if url.startswith("/"): | ||||||
|  |         return "https://www.facebook.com" + url | ||||||
|  |     return url | ||||||
							
								
								
									
										1038
									
								
								fbchat/client.py
									
									
									
									
									
								
							
							
						
						| @@ -1,31 +1,29 @@ | |||||||
|  | # -*- coding: UTF-8 -*- | ||||||
|  | """This file is here to maintain backwards compatability, and to re-export our models | ||||||
|  | into the global module (see `__init__.py`). | ||||||
|  |  | ||||||
|  | A common pattern was to use `from fbchat.models import *`, hence we need this while | ||||||
|  | transitioning to a better code structure. | ||||||
|  | """ | ||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
| import sys |  | ||||||
|  |  | ||||||
| class Base(): | from ._core import Enum | ||||||
|     def __repr__(self): | from ._exception import FBchatException, FBchatFacebookError, FBchatUserError | ||||||
|         uni = self.__unicode__() | from ._thread import ThreadType, ThreadLocation, ThreadColor, Thread | ||||||
|         return uni.encode('utf-8') if sys.version_info < (3, 0) else uni | from ._user import TypingStatus, User, ActiveStatus | ||||||
|  | from ._group import Group, Room | ||||||
|     def __unicode__(self): | from ._page import Page | ||||||
|         return u'<%s %s (%s)>' % (self.type.upper(), self.name, self.url) | from ._message import EmojiSize, MessageReaction, Mention, Message | ||||||
|  | from ._attachment import Attachment, UnsentMessage, ShareAttachment | ||||||
| class User(Base): | from ._sticker import Sticker | ||||||
|     def __init__(self, data): | from ._location import LocationAttachment, LiveLocationAttachment | ||||||
|         if data['type'] != 'user': | from ._file import FileAttachment, AudioAttachment, ImageAttachment, VideoAttachment | ||||||
|             raise Exception("[!] %s <%s> is not a user" % (data['text'], data['path'])) | from ._quick_reply import ( | ||||||
|         self.uid = data['uid'] |     QuickReply, | ||||||
|         self.type = data['type'] |     QuickReplyText, | ||||||
|         self.photo = data['photo'] |     QuickReplyLocation, | ||||||
|         self.url = data['path'] |     QuickReplyPhoneNumber, | ||||||
|         self.name = data['text'] |     QuickReplyEmail, | ||||||
|         self.score = data['score'] | ) | ||||||
|  | from ._poll import Poll, PollOption | ||||||
|         self.data = data | from ._plan import GuestStatus, Plan | ||||||
|  |  | ||||||
| class Thread(): |  | ||||||
|     def __init__(self, **entries):  |  | ||||||
|         self.__dict__.update(entries) |  | ||||||
|  |  | ||||||
| class Message(): |  | ||||||
|     def __init__(self, **entries): |  | ||||||
|         self.__dict__.update(entries) |  | ||||||
|   | |||||||
| @@ -1,8 +0,0 @@ | |||||||
| LIKES={ |  | ||||||
|     'l': '369239383222810', |  | ||||||
|     'm': '369239343222814', |  | ||||||
|     's': '369239263222822' |  | ||||||
| } |  | ||||||
| LIKES['large'] = LIKES['l'] |  | ||||||
| LIKES['medium'] =LIKES['m'] |  | ||||||
| LIKES['small'] = LIKES['s'] |  | ||||||
| @@ -1,64 +0,0 @@ | |||||||
| import re |  | ||||||
| import json |  | ||||||
| from time import time |  | ||||||
| from random import random |  | ||||||
| USER_AGENTS = [ |  | ||||||
|     "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/42.0.2311.90 Safari/537.36", |  | ||||||
|     "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_3) AppleWebKit/601.1.10 (KHTML, like Gecko) Version/8.0.5 Safari/601.1.10", |  | ||||||
|     "Mozilla/5.0 (Windows NT 6.3; WOW64; ; NCT50_AAP285C84A1328) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/42.0.2311.90 Safari/537.36", |  | ||||||
|     "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.1 (KHTML, like Gecko) Chrome/22.0.1207.1 Safari/537.1", |  | ||||||
|     "Mozilla/5.0 (X11; CrOS i686 2268.111.0) AppleWebKit/536.11 (KHTML, like Gecko) Chrome/20.0.1132.57 Safari/536.11", |  | ||||||
|     "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/536.6 (KHTML, like Gecko) Chrome/20.0.1092.0 Safari/536.6" |  | ||||||
| ] |  | ||||||
| GENDERS = { |  | ||||||
|     0: 'unknown', |  | ||||||
|     1: 'female_singular', |  | ||||||
|     2: 'male_singular', |  | ||||||
|     3: 'female_singular_guess', |  | ||||||
|     4: 'male_singular_guess', |  | ||||||
|     5: 'mixed', |  | ||||||
|     6: 'neuter_singular', |  | ||||||
|     7: 'unknown_singular', |  | ||||||
|     8: 'female_plural', |  | ||||||
|     9: 'male_plural', |  | ||||||
|     10: 'neuter_plural', |  | ||||||
|     11: 'unknown_plural', |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def now(): |  | ||||||
|     return int(time()*1000) |  | ||||||
|  |  | ||||||
| def strip_to_json(text): |  | ||||||
|     return text[text.index('{'):] |  | ||||||
|  |  | ||||||
| def get_json(text): |  | ||||||
|     return json.loads(strip_to_json(text)) |  | ||||||
|  |  | ||||||
| def digit_to_char(digit): |  | ||||||
|     if digit < 10: |  | ||||||
|         return str(digit) |  | ||||||
|     return chr(ord('a') + digit - 10) |  | ||||||
|  |  | ||||||
| def str_base(number,base): |  | ||||||
|     if number < 0: |  | ||||||
|         return '-' + str_base(-number, base) |  | ||||||
|     (d, m) = divmod(number, base) |  | ||||||
|     if d > 0: |  | ||||||
|         return str_base(d, base) + digit_to_char(m) |  | ||||||
|     return digit_to_char(m) |  | ||||||
|  |  | ||||||
| def generateMessageID(client_id=None): |  | ||||||
|     k = now() |  | ||||||
|     l = int(random() * 4294967295) |  | ||||||
|     return ("<%s:%s-%s@mail.projektitan.com>" % (k, l, client_id)); |  | ||||||
|  |  | ||||||
| def getSignatureID(): |  | ||||||
|     return hex(int(random() * 2147483648)) |  | ||||||
|  |  | ||||||
| def generateOfflineThreadingID() : |  | ||||||
|     ret = now() |  | ||||||
|     value = int(random() * 4294967295); |  | ||||||
|     string = ("0000000000000000000000" + bin(value))[-22:] |  | ||||||
|     msgs = bin(ret) + string |  | ||||||
|     return str(int(msgs,2)) |  | ||||||
|  |  | ||||||
							
								
								
									
										68
									
								
								pyproject.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,68 @@ | |||||||
|  | [tool.black] | ||||||
|  | line-length = 88 | ||||||
|  |  | ||||||
|  | [build-system] | ||||||
|  | requires = ["flit"] | ||||||
|  | build-backend = "flit.buildapi" | ||||||
|  |  | ||||||
|  | [tool.flit.metadata] | ||||||
|  | module = "fbchat" | ||||||
|  | author = "Taehoon Kim" | ||||||
|  | author-email = "carpedm20@gmail.com" | ||||||
|  | maintainer = "Mads Marquart" | ||||||
|  | maintainer-email = "madsmtm@gmail.com" | ||||||
|  | home-page = "https://github.com/carpedm20/fbchat/" | ||||||
|  | requires = [ | ||||||
|  |     "aenum~=2.0", | ||||||
|  |     "attrs>=18.2", | ||||||
|  |     "requests~=2.19", | ||||||
|  |     "beautifulsoup4~=4.0", | ||||||
|  |     "paho-mqtt~=1.5", | ||||||
|  | ] | ||||||
|  | description-file = "README.rst" | ||||||
|  | classifiers = [ | ||||||
|  |     "Development Status :: 3 - Alpha", | ||||||
|  |     "Intended Audience :: Developers", | ||||||
|  |     "Intended Audience :: Information Technology", | ||||||
|  |     "License :: OSI Approved :: BSD License", | ||||||
|  |     "Operating System :: OS Independent", | ||||||
|  |     "Natural Language :: English", | ||||||
|  |     "Programming Language :: Python", | ||||||
|  |     "Programming Language :: Python :: 2.7", | ||||||
|  |     "Programming Language :: Python :: 3", | ||||||
|  |     "Programming Language :: Python :: 3.4", | ||||||
|  |     "Programming Language :: Python :: 3.5", | ||||||
|  |     "Programming Language :: Python :: 3.6", | ||||||
|  |     "Programming Language :: Python :: 3.7", | ||||||
|  |     "Programming Language :: Python :: Implementation :: CPython", | ||||||
|  |     "Programming Language :: Python :: Implementation :: PyPy", | ||||||
|  |     "Topic :: Communications :: Chat", | ||||||
|  |     "Topic :: Internet :: WWW/HTTP", | ||||||
|  |     "Topic :: Internet :: WWW/HTTP :: Dynamic Content", | ||||||
|  |     "Topic :: Software Development :: Libraries", | ||||||
|  |     "Topic :: Software Development :: Libraries :: Python Modules", | ||||||
|  | ] | ||||||
|  | requires-python = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, <4.0" | ||||||
|  | keywords = "Facebook FB Messenger Library Chat Api Bot" | ||||||
|  | license = "BSD 3-Clause" | ||||||
|  |  | ||||||
|  | [tool.flit.metadata.urls] | ||||||
|  | Documentation = "https://fbchat.readthedocs.io/" | ||||||
|  | Repository = "https://github.com/carpedm20/fbchat/" | ||||||
|  |  | ||||||
|  | [tool.flit.metadata.requires-extra] | ||||||
|  | test = [ | ||||||
|  |     "pytest~=4.0", | ||||||
|  |     "six~=1.0", | ||||||
|  | ] | ||||||
|  | docs = [ | ||||||
|  |     "sphinx~=2.0", | ||||||
|  |     "sphinxcontrib-spelling~=4.0" | ||||||
|  | ] | ||||||
|  | lint = [ | ||||||
|  |     "black", | ||||||
|  | ] | ||||||
|  | tools = [ | ||||||
|  |     # Fork of bumpversion, see https://github.com/c4urself/bump2version | ||||||
|  |     "bump2version~=0.5.0", | ||||||
|  | ] | ||||||
							
								
								
									
										6
									
								
								pytest.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | |||||||
|  | [pytest] | ||||||
|  | xfail_strict=true | ||||||
|  | markers = | ||||||
|  |     offline: Offline tests, aka. tests that can be executed without the need of a client | ||||||
|  |     expensive: Expensive tests, which should be executed sparingly | ||||||
|  | addopts = -m "not expensive" | ||||||
							
								
								
									
										78
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						| @@ -1,78 +0,0 @@ | |||||||
| #!/usr/bin/env python |  | ||||||
|  |  | ||||||
|  |  | ||||||
| """ |  | ||||||
| Setup script for fbchat |  | ||||||
| """ |  | ||||||
|  |  | ||||||
|  |  | ||||||
| import os |  | ||||||
| try: |  | ||||||
|     from setuptools import setup |  | ||||||
| except ImportError: |  | ||||||
|     from distutils.core import setup |  | ||||||
|  |  | ||||||
|  |  | ||||||
| with open('README.rst') as f: |  | ||||||
|     readme_content = f.read().strip() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| version = None |  | ||||||
| author = None |  | ||||||
| email = None |  | ||||||
| source = None |  | ||||||
| with open(os.path.join('fbchat', '__init__.py')) as f: |  | ||||||
|     for line in f: |  | ||||||
|         if line.strip().startswith('__version__'): |  | ||||||
|             version = line.split('=')[1].strip().replace('"', '').replace("'", '') |  | ||||||
|         elif line.strip().startswith('__author__'): |  | ||||||
|             author = line.split('=')[1].strip().replace('"', '').replace("'", '') |  | ||||||
|         elif line.strip().startswith('__email__'): |  | ||||||
|             email = line.split('=')[1].strip().replace('"', '').replace("'", '') |  | ||||||
|         elif line.strip().startswith('__source__'): |  | ||||||
|             source = line.split('=')[1].strip().replace('"', '').replace("'", '') |  | ||||||
|         elif None not in (version, author, email, source): |  | ||||||
|             break |  | ||||||
|  |  | ||||||
| setup( |  | ||||||
|     name='fbchat', |  | ||||||
|     author=author, |  | ||||||
|     author_email=email, |  | ||||||
|     license='BSD License', |  | ||||||
|     keywords=["facebook chat fbchat"], |  | ||||||
|     description="Facebook Chat (Messenger) for Python", |  | ||||||
|     long_description=readme_content, |  | ||||||
|     classifiers=[ |  | ||||||
|         'Development Status :: 2 - Pre-Alpha', |  | ||||||
|         'Intended Audience :: Developers', |  | ||||||
|         'Intended Audience :: Developers', |  | ||||||
|         'Intended Audience :: Information Technology', |  | ||||||
|         'License :: OSI Approved :: BSD License', |  | ||||||
|         'License :: OSI Approved :: BSD License', |  | ||||||
|         'Operating System :: OS Independent', |  | ||||||
|         'Programming Language :: Python :: 2.6', |  | ||||||
|         'Programming Language :: Python :: 2.7', |  | ||||||
|         'Programming Language :: Python :: 2.7', |  | ||||||
|         'Programming Language :: Python :: 3.2', |  | ||||||
|         'Programming Language :: Python :: 3.3', |  | ||||||
|         'Programming Language :: Python :: 3.3', |  | ||||||
|         'Programming Language :: Python :: 3.4', |  | ||||||
|         'Programming Language :: Python :: 3.5', |  | ||||||
|         'Programming Language :: Python :: Implementation :: CPython', |  | ||||||
|         'Programming Language :: Python :: Implementation :: PyPy', |  | ||||||
|         'Programming Language :: Python', |  | ||||||
|         'Topic :: Software Development :: Libraries :: Python Modules', |  | ||||||
|         'Topic :: Internet :: WWW/HTTP :: Dynamic Content', |  | ||||||
|         'Topic :: Communications :: Chat', |  | ||||||
|     ], |  | ||||||
|     include_package_data=True, |  | ||||||
|     packages=['fbchat'], |  | ||||||
|     install_requires=[ |  | ||||||
|         'requests', |  | ||||||
|         'lxml', |  | ||||||
|         'beautifulsoup4' |  | ||||||
|     ], |  | ||||||
|     url=source, |  | ||||||
|     version=version, |  | ||||||
|     zip_safe=True, |  | ||||||
| ) |  | ||||||
							
								
								
									
										168
									
								
								tests.py
									
									
									
									
									
								
							
							
						
						| @@ -1,168 +0,0 @@ | |||||||
| #!/usr/bin/env python |  | ||||||
|  |  | ||||||
| import logging |  | ||||||
| import fbchat |  | ||||||
| import getpass |  | ||||||
| import unittest |  | ||||||
| import sys |  | ||||||
| from os import path |  | ||||||
|  |  | ||||||
| # Disable logging |  | ||||||
| logging.basicConfig(level=100) |  | ||||||
| fbchat.log.setLevel(100) |  | ||||||
|  |  | ||||||
| """ |  | ||||||
|  |  | ||||||
| Tests for fbchat |  | ||||||
| ~~~~~~~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| To use these tests, put: |  | ||||||
| - email |  | ||||||
| - password |  | ||||||
| - a group_uid |  | ||||||
| - a user_uid (the user will be kicked from the group and then added again) |  | ||||||
| (seperated these by a newline) in a file called `tests.data`, or type them manually in the terminal prompts |  | ||||||
|  |  | ||||||
| Please remember to test both python v. 2.7 and python v. 3.6! |  | ||||||
|  |  | ||||||
| If you've made any changes to the 2FA functionality, test it with a 2FA enabled account |  | ||||||
| If you only want to execute specific tests, pass the function names in the commandline |  | ||||||
|  |  | ||||||
| """ |  | ||||||
|  |  | ||||||
| class TestFbchat(unittest.TestCase): |  | ||||||
|     def test_login_functions(self): |  | ||||||
|         self.assertTrue(client.is_logged_in()) |  | ||||||
|          |  | ||||||
|         client.logout() |  | ||||||
|          |  | ||||||
|         self.assertFalse(client.is_logged_in()) |  | ||||||
|          |  | ||||||
|         with self.assertRaises(Exception): |  | ||||||
|             client.login("not@email.com", "not_password", max_retries=1) |  | ||||||
|          |  | ||||||
|         client.login(email, password) |  | ||||||
|          |  | ||||||
|         self.assertTrue(client.is_logged_in()) |  | ||||||
|  |  | ||||||
|     def test_sessions(self): |  | ||||||
|         global client |  | ||||||
|         session_cookies = client.getSession() |  | ||||||
|         client = fbchat.Client(email, password, session_cookies=session_cookies) |  | ||||||
|          |  | ||||||
|         self.assertTrue(client.is_logged_in()) |  | ||||||
|  |  | ||||||
|     def test_setDefaultRecipient(self): |  | ||||||
|         client.setDefaultRecipient(client.uid, is_user=True) |  | ||||||
|         self.assertTrue(client.send(message="test_default_recipient")) |  | ||||||
|  |  | ||||||
|     def test_getAllUsers(self): |  | ||||||
|         users = client.getAllUsers() |  | ||||||
|         self.assertGreater(len(users), 0) |  | ||||||
|  |  | ||||||
|     def test_getUsers(self): |  | ||||||
|         users = client.getUsers("Mark Zuckerberg") |  | ||||||
|         self.assertGreater(len(users), 0) |  | ||||||
|          |  | ||||||
|         u = users[0] |  | ||||||
|          |  | ||||||
|         # Test if values are set correctly |  | ||||||
|         self.assertIsInstance(u.uid, int) |  | ||||||
|         self.assertEquals(u.type, 'user') |  | ||||||
|         self.assertEquals(u.photo[:4], 'http') |  | ||||||
|         self.assertEquals(u.url[:4], 'http') |  | ||||||
|         self.assertEquals(u.name, 'Mark Zuckerberg') |  | ||||||
|         self.assertGreater(u.score, 0) |  | ||||||
|      |  | ||||||
|     def test_send_likes(self): |  | ||||||
|         self.assertTrue(client.send(client.uid, like='s')) |  | ||||||
|         self.assertTrue(client.send(client.uid, like='m')) |  | ||||||
|         self.assertTrue(client.send(client.uid, like='l')) |  | ||||||
|         self.assertTrue(client.send(group_uid, like='s', is_user=False)) |  | ||||||
|         self.assertTrue(client.send(group_uid, like='m', is_user=False)) |  | ||||||
|         self.assertTrue(client.send(group_uid, like='l', is_user=False)) |  | ||||||
|      |  | ||||||
|     def test_send(self): |  | ||||||
|         self.assertTrue(client.send(client.uid, message='test_send_user')) |  | ||||||
|         self.assertTrue(client.send(group_uid, message='test_send_group', is_user=False)) |  | ||||||
|      |  | ||||||
|     def test_send_images(self): |  | ||||||
|         image_url = 'https://cdn4.iconfinder.com/data/icons/ionicons/512/icon-image-128.png' |  | ||||||
|         image_local_url = path.join(path.dirname(__file__), 'test_image.png') |  | ||||||
|         self.assertTrue(client.sendRemoteImage(client.uid, message='test_send_user_images_remote', image=image_url)) |  | ||||||
|         self.assertTrue(client.sendLocalImage(client.uid, message='test_send_user_images_local', image=image_local_url)) |  | ||||||
|         self.assertTrue(client.sendRemoteImage(group_uid, message='test_send_group_images_remote', is_user=False, image=image_url)) |  | ||||||
|         self.assertTrue(client.sendLocalImage(group_uid, message='test_send_group_images_local', is_user=False, image=image_local_url)) |  | ||||||
|      |  | ||||||
|     def test_getThreadInfo(self): |  | ||||||
|         info = client.getThreadInfo(client.uid, last_n=1) |  | ||||||
|         self.assertEquals(info[0].author, 'fbid:' + str(client.uid)) |  | ||||||
|         client.send(group_uid, message='test_getThreadInfo', is_user=False) |  | ||||||
|         info = client.getThreadInfo(group_uid, last_n=1, is_user=False) |  | ||||||
|         self.assertEquals(info[0].author, 'fbid:' + str(client.uid)) |  | ||||||
|         self.assertEquals(info[0].body, 'test_getThreadInfo') |  | ||||||
|  |  | ||||||
|     def test_markAs(self): |  | ||||||
|         # To be implemented (requires some form of manual watching) |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     def test_listen(self): |  | ||||||
|         client.do_one_listen() |  | ||||||
|  |  | ||||||
|     def test_getUserInfo(self): |  | ||||||
|         info = client.getUserInfo(4) |  | ||||||
|         self.assertEquals(info['name'], 'Mark Zuckerberg') |  | ||||||
|      |  | ||||||
|     def test_remove_add_from_chat(self): |  | ||||||
|         self.assertTrue(client.remove_user_from_chat(group_uid, user_uid)) |  | ||||||
|         self.assertTrue(client.add_users_to_chat(group_uid, user_uid)) |  | ||||||
|      |  | ||||||
|     def test_changeThreadTitle(self): |  | ||||||
|         self.assertTrue(client.changeThreadTitle(group_uid, 'test_changeThreadTitle')) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def start_test(param_client, param_group_uid, param_user_uid, tests=[]): |  | ||||||
|     global client |  | ||||||
|     global group_uid |  | ||||||
|     global user_uid |  | ||||||
|      |  | ||||||
|     client = param_client |  | ||||||
|     group_uid = param_group_uid |  | ||||||
|     user_uid = param_user_uid |  | ||||||
|      |  | ||||||
|     if len(tests) == 0: |  | ||||||
|         suite = unittest.TestLoader().loadTestsFromTestCase(TestFbchat) |  | ||||||
|     else: |  | ||||||
|         suite = unittest.TestSuite(map(TestFbchat, tests)) |  | ||||||
|     print ('Starting test(s)') |  | ||||||
|     unittest.TextTestRunner(verbosity=2).run(suite) |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': |  | ||||||
|     # Python 3 does not use raw_input, whereas Python 2 does |  | ||||||
|     try: |  | ||||||
|         input = raw_input |  | ||||||
|     except Exception as e: |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|     try: |  | ||||||
|         with open(path.join(path.dirname(__file__), 'tests.data'), 'r') as f: |  | ||||||
|             content = f.readlines() |  | ||||||
|         content = [x.strip() for x in content if len(x.strip()) != 0] |  | ||||||
|         email = content[0] |  | ||||||
|         password = content[1] |  | ||||||
|         group_uid = content[2] |  | ||||||
|         user_uid = content[3] |  | ||||||
|     except (IOError, IndexError) as e: |  | ||||||
|         email = input('Email: ') |  | ||||||
|         password = getpass.getpass() |  | ||||||
|         group_uid = input('Please enter a group uid (To test group functionality): ') |  | ||||||
|         user_uid = input('Please enter a user uid (To test kicking/adding functionality): ') |  | ||||||
|  |  | ||||||
|     print ('Logging in') |  | ||||||
|     client = fbchat.Client(email, password) |  | ||||||
|      |  | ||||||
|     # Warning! Taking user input directly like this could be dangerous! Use only for testing purposes! |  | ||||||
|     start_test(client, group_uid, user_uid, sys.argv[1:]) |  | ||||||
|  |  | ||||||
							
								
								
									
										129
									
								
								tests/conftest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,129 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  | import json | ||||||
|  |  | ||||||
|  | from utils import * | ||||||
|  | from contextlib import contextmanager | ||||||
|  | from fbchat.models import ThreadType, Message, Mention | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(scope="session") | ||||||
|  | def user(client2): | ||||||
|  |     return {"id": client2.uid, "type": ThreadType.USER} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(scope="session") | ||||||
|  | def group(pytestconfig): | ||||||
|  |     return { | ||||||
|  |         "id": load_variable("group_id", pytestconfig.cache), | ||||||
|  |         "type": ThreadType.GROUP, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture( | ||||||
|  |     scope="session", | ||||||
|  |     params=["user", "group", pytest.param("none", marks=[pytest.mark.xfail()])], | ||||||
|  | ) | ||||||
|  | def thread(request, user, group): | ||||||
|  |     return { | ||||||
|  |         "user": user, | ||||||
|  |         "group": group, | ||||||
|  |         "none": {"id": "0", "type": ThreadType.GROUP}, | ||||||
|  |     }[request.param] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(scope="session") | ||||||
|  | def client1(pytestconfig): | ||||||
|  |     with load_client(1, pytestconfig.cache) as c: | ||||||
|  |         yield c | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(scope="session") | ||||||
|  | def client2(pytestconfig): | ||||||
|  |     with load_client(2, pytestconfig.cache) as c: | ||||||
|  |         yield c | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(scope="module") | ||||||
|  | def client(client1, thread): | ||||||
|  |     client1.setDefaultThread(thread["id"], thread["type"]) | ||||||
|  |     yield client1 | ||||||
|  |     client1.resetDefaultThread() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(scope="session", params=["client1", "client2"]) | ||||||
|  | def client_all(request, client1, client2): | ||||||
|  |     return client1 if request.param == "client1" else client2 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(scope="session") | ||||||
|  | def catch_event(client2): | ||||||
|  |     t = ClientThread(client2) | ||||||
|  |     t.start() | ||||||
|  |  | ||||||
|  |     @contextmanager | ||||||
|  |     def inner(method_name): | ||||||
|  |         caught = CaughtValue() | ||||||
|  |         old_method = getattr(client2, method_name) | ||||||
|  |  | ||||||
|  |         # Will be called by the other thread | ||||||
|  |         def catch_value(*args, **kwargs): | ||||||
|  |             old_method(*args, **kwargs) | ||||||
|  |             # Make sure the `set` is only called once | ||||||
|  |             if not caught.is_set(): | ||||||
|  |                 caught.set(kwargs) | ||||||
|  |  | ||||||
|  |         setattr(client2, method_name, catch_value) | ||||||
|  |         yield caught | ||||||
|  |         caught.wait() | ||||||
|  |         if not caught.is_set(): | ||||||
|  |             raise ValueError("The value could not be caught") | ||||||
|  |         setattr(client2, method_name, old_method) | ||||||
|  |  | ||||||
|  |     yield inner | ||||||
|  |  | ||||||
|  |     t.should_stop.set() | ||||||
|  |  | ||||||
|  |     try: | ||||||
|  |         # Make the client send a messages to itself, so the blocking pull request will return | ||||||
|  |         # This is probably not safe, since the client is making two requests simultaneously | ||||||
|  |         client2.sendMessage(random_hex(), client2.uid) | ||||||
|  |     finally: | ||||||
|  |         t.join() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(scope="module") | ||||||
|  | def compare(client, thread): | ||||||
|  |     def inner(caught_event, **kwargs): | ||||||
|  |         d = { | ||||||
|  |             "author_id": client.uid, | ||||||
|  |             "thread_id": client.uid | ||||||
|  |             if thread["type"] == ThreadType.USER | ||||||
|  |             else thread["id"], | ||||||
|  |             "thread_type": thread["type"], | ||||||
|  |         } | ||||||
|  |         d.update(kwargs) | ||||||
|  |         return subset(caught_event.res, **d) | ||||||
|  |  | ||||||
|  |     return inner | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(params=["me", "other", "me other"]) | ||||||
|  | def message_with_mentions(request, client, client2, group): | ||||||
|  |     text = "Hi there [" | ||||||
|  |     mentions = [] | ||||||
|  |     if "me" in request.param: | ||||||
|  |         mentions.append(Mention(thread_id=client.uid, offset=len(text), length=2)) | ||||||
|  |         text += "me, " | ||||||
|  |     if "other" in request.param: | ||||||
|  |         mentions.append(Mention(thread_id=client2.uid, offset=len(text), length=5)) | ||||||
|  |         text += "other, " | ||||||
|  |     # Unused, because Facebook don't properly support sending mentions with groups as targets | ||||||
|  |     if "group" in request.param: | ||||||
|  |         mentions.append(Mention(thread_id=group["id"], offset=len(text), length=5)) | ||||||
|  |         text += "group, " | ||||||
|  |     text += "nothing]" | ||||||
|  |     return Message(text, mentions=mentions) | ||||||
							
								
								
									
										
											BIN
										
									
								
								tests/resources/audio.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										4
									
								
								tests/resources/file.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | |||||||
|  | { | ||||||
|  |     "some": "data", | ||||||
|  |     "in": "here" | ||||||
|  | } | ||||||
							
								
								
									
										1
									
								
								tests/resources/file.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | This is just a text file | ||||||
							
								
								
									
										
											BIN
										
									
								
								tests/resources/image.gif
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.3 KiB | 
							
								
								
									
										
											BIN
										
									
								
								tests/resources/image.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.6 KiB | 
| Before Width: | Height: | Size: 1.6 KiB After Width: | Height: | Size: 1.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								tests/resources/video.mp4
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										55
									
								
								tests/test_base.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,55 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  | import py_compile | ||||||
|  |  | ||||||
|  | from glob import glob | ||||||
|  | from os import path, environ | ||||||
|  | from fbchat import Client | ||||||
|  | from fbchat.models import FBchatUserError, Message | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.offline | ||||||
|  | def test_examples(): | ||||||
|  |     # Compiles the examples, to check for syntax errors | ||||||
|  |     for name in glob(path.join(path.dirname(__file__), "../examples", "*.py")): | ||||||
|  |         py_compile.compile(name) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.trylast | ||||||
|  | @pytest.mark.expensive | ||||||
|  | def test_login(client1): | ||||||
|  |     assert client1.isLoggedIn() | ||||||
|  |     email = client1.email | ||||||
|  |     password = client1.password | ||||||
|  |  | ||||||
|  |     client1.logout() | ||||||
|  |  | ||||||
|  |     assert not client1.isLoggedIn() | ||||||
|  |  | ||||||
|  |     with pytest.raises(FBchatUserError): | ||||||
|  |         client1.login("<invalid email>", "<invalid password>", max_tries=1) | ||||||
|  |  | ||||||
|  |     client1.login(email, password) | ||||||
|  |  | ||||||
|  |     assert client1.isLoggedIn() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.trylast | ||||||
|  | def test_sessions(client1): | ||||||
|  |     session = client1.getSession() | ||||||
|  |     Client("no email needed", "no password needed", session_cookies=session) | ||||||
|  |     client1.setSession(session) | ||||||
|  |     assert client1.isLoggedIn() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.tryfirst | ||||||
|  | def test_default_thread(client1, thread): | ||||||
|  |     client1.setDefaultThread(thread["id"], thread["type"]) | ||||||
|  |     assert client1.send(Message(text="Sent to the specified thread")) | ||||||
|  |  | ||||||
|  |     client1.resetDefaultThread() | ||||||
|  |     with pytest.raises(ValueError): | ||||||
|  |         client1.send(Message(text="Should not be sent")) | ||||||
							
								
								
									
										101
									
								
								tests/test_fetch.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,101 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from os import path | ||||||
|  | from fbchat.models import ThreadType, Message, Mention, EmojiSize, Sticker | ||||||
|  | from utils import subset, STICKER_LIST, EMOJI_LIST | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_fetch_all_users(client1): | ||||||
|  |     users = client1.fetchAllUsers() | ||||||
|  |     assert len(users) > 0 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_fetch_thread_list(client1): | ||||||
|  |     threads = client1.fetchThreadList(limit=2) | ||||||
|  |     assert len(threads) == 2 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_fetch_threads(client1): | ||||||
|  |     threads = client1.fetchThreads(limit=2) | ||||||
|  |     assert len(threads) == 2 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("emoji, emoji_size", EMOJI_LIST) | ||||||
|  | def test_fetch_message_emoji(client, emoji, emoji_size): | ||||||
|  |     mid = client.sendEmoji(emoji, emoji_size) | ||||||
|  |     (message,) = client.fetchThreadMessages(limit=1) | ||||||
|  |  | ||||||
|  |     assert subset( | ||||||
|  |         vars(message), uid=mid, author=client.uid, text=emoji, emoji_size=emoji_size | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("emoji, emoji_size", EMOJI_LIST) | ||||||
|  | def test_fetch_message_info_emoji(client, thread, emoji, emoji_size): | ||||||
|  |     mid = client.sendEmoji(emoji, emoji_size) | ||||||
|  |     message = client.fetchMessageInfo(mid, thread_id=thread["id"]) | ||||||
|  |  | ||||||
|  |     assert subset( | ||||||
|  |         vars(message), uid=mid, author=client.uid, text=emoji, emoji_size=emoji_size | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_fetch_message_mentions(client, thread, message_with_mentions): | ||||||
|  |     mid = client.send(message_with_mentions) | ||||||
|  |     (message,) = client.fetchThreadMessages(limit=1) | ||||||
|  |  | ||||||
|  |     assert subset( | ||||||
|  |         vars(message), uid=mid, author=client.uid, text=message_with_mentions.text | ||||||
|  |     ) | ||||||
|  |     # The mentions are not ordered by offset | ||||||
|  |     for m in message.mentions: | ||||||
|  |         assert vars(m) in [vars(x) for x in message_with_mentions.mentions] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_fetch_message_info_mentions(client, thread, message_with_mentions): | ||||||
|  |     mid = client.send(message_with_mentions) | ||||||
|  |     message = client.fetchMessageInfo(mid, thread_id=thread["id"]) | ||||||
|  |  | ||||||
|  |     assert subset( | ||||||
|  |         vars(message), uid=mid, author=client.uid, text=message_with_mentions.text | ||||||
|  |     ) | ||||||
|  |     # The mentions are not ordered by offset | ||||||
|  |     for m in message.mentions: | ||||||
|  |         assert vars(m) in [vars(x) for x in message_with_mentions.mentions] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("sticker", STICKER_LIST) | ||||||
|  | def test_fetch_message_sticker(client, sticker): | ||||||
|  |     mid = client.send(Message(sticker=sticker)) | ||||||
|  |     (message,) = client.fetchThreadMessages(limit=1) | ||||||
|  |  | ||||||
|  |     assert subset(vars(message), uid=mid, author=client.uid) | ||||||
|  |     assert subset(vars(message.sticker), uid=sticker.uid) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("sticker", STICKER_LIST) | ||||||
|  | def test_fetch_message_info_sticker(client, thread, sticker): | ||||||
|  |     mid = client.send(Message(sticker=sticker)) | ||||||
|  |     message = client.fetchMessageInfo(mid, thread_id=thread["id"]) | ||||||
|  |  | ||||||
|  |     assert subset(vars(message), uid=mid, author=client.uid) | ||||||
|  |     assert subset(vars(message.sticker), uid=sticker.uid) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_fetch_info(client1, group): | ||||||
|  |     info = client1.fetchUserInfo("4")["4"] | ||||||
|  |     assert info.name == "Mark Zuckerberg" | ||||||
|  |  | ||||||
|  |     info = client1.fetchGroupInfo(group["id"])[group["id"]] | ||||||
|  |     assert info.type == ThreadType.GROUP | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_fetch_image_url(client): | ||||||
|  |     client.sendLocalFiles([path.join(path.dirname(__file__), "resources", "image.png")]) | ||||||
|  |     (message,) = client.fetchThreadMessages(limit=1) | ||||||
|  |  | ||||||
|  |     assert client.fetchImageUrl(message.attachments[0].uid) | ||||||
							
								
								
									
										23
									
								
								tests/test_message_management.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from fbchat.models import Message, MessageReaction | ||||||
|  | from utils import subset | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_set_reaction(client): | ||||||
|  |     mid = client.send(Message(text="This message will be reacted to")) | ||||||
|  |     client.reactToMessage(mid, MessageReaction.LOVE) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_delete_messages(client): | ||||||
|  |     text1 = "This message will stay" | ||||||
|  |     text2 = "This message will be removed" | ||||||
|  |     mid1 = client.sendMessage(text1) | ||||||
|  |     mid2 = client.sendMessage(text2) | ||||||
|  |     client.deleteMessages(mid2) | ||||||
|  |     (message,) = client.fetchThreadMessages(limit=1) | ||||||
|  |     assert subset(vars(message), uid=mid1, author=client.uid, text=text1) | ||||||
							
								
								
									
										114
									
								
								tests/test_plans.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,114 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from fbchat.models import Plan, FBchatFacebookError, ThreadType | ||||||
|  | from utils import random_hex, subset | ||||||
|  | from time import time | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture( | ||||||
|  |     scope="module", | ||||||
|  |     params=[ | ||||||
|  |         Plan(int(time()) + 100, random_hex()), | ||||||
|  |         pytest.param( | ||||||
|  |             Plan(int(time()), random_hex()), | ||||||
|  |             marks=[pytest.mark.xfail(raises=FBchatFacebookError)], | ||||||
|  |         ), | ||||||
|  |         pytest.param(Plan(0, None), marks=[pytest.mark.xfail()]), | ||||||
|  |     ], | ||||||
|  | ) | ||||||
|  | def plan_data(request, client, user, thread, catch_event, compare): | ||||||
|  |     with catch_event("onPlanCreated") as x: | ||||||
|  |         client.createPlan(request.param, thread["id"]) | ||||||
|  |     assert compare(x) | ||||||
|  |     assert subset( | ||||||
|  |         vars(x.res["plan"]), | ||||||
|  |         time=request.param.time, | ||||||
|  |         title=request.param.title, | ||||||
|  |         author_id=client.uid, | ||||||
|  |         going=[client.uid], | ||||||
|  |         declined=[], | ||||||
|  |     ) | ||||||
|  |     plan_id = x.res["plan"] | ||||||
|  |     assert user["id"] in x.res["plan"].invited | ||||||
|  |     request.param.uid = x.res["plan"].uid | ||||||
|  |     yield x.res, request.param | ||||||
|  |     with catch_event("onPlanDeleted") as x: | ||||||
|  |         client.deletePlan(plan_id) | ||||||
|  |     assert compare(x) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.tryfirst | ||||||
|  | def test_create_delete_plan(plan_data): | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_fetch_plan_info(client, catch_event, plan_data): | ||||||
|  |     event, plan = plan_data | ||||||
|  |     fetched_plan = client.fetchPlanInfo(plan.uid) | ||||||
|  |     assert subset( | ||||||
|  |         vars(fetched_plan), time=plan.time, title=plan.title, author_id=int(client.uid) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("take_part", [False, True]) | ||||||
|  | def test_change_plan_participation( | ||||||
|  |     client, thread, catch_event, compare, plan_data, take_part | ||||||
|  | ): | ||||||
|  |     event, plan = plan_data | ||||||
|  |     with catch_event("onPlanParticipation") as x: | ||||||
|  |         client.changePlanParticipation(plan, take_part=take_part) | ||||||
|  |     assert compare(x, take_part=take_part) | ||||||
|  |     assert subset( | ||||||
|  |         vars(x.res["plan"]), | ||||||
|  |         time=plan.time, | ||||||
|  |         title=plan.title, | ||||||
|  |         author_id=client.uid, | ||||||
|  |         going=[client.uid] if take_part else [], | ||||||
|  |         declined=[client.uid] if not take_part else [], | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.trylast | ||||||
|  | def test_edit_plan(client, thread, catch_event, compare, plan_data): | ||||||
|  |     event, plan = plan_data | ||||||
|  |     new_plan = Plan(plan.time + 100, random_hex()) | ||||||
|  |     with catch_event("onPlanEdited") as x: | ||||||
|  |         client.editPlan(plan, new_plan) | ||||||
|  |     assert compare(x) | ||||||
|  |     assert subset( | ||||||
|  |         vars(x.res["plan"]), | ||||||
|  |         time=new_plan.time, | ||||||
|  |         title=new_plan.title, | ||||||
|  |         author_id=client.uid, | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.trylast | ||||||
|  | @pytest.mark.expensive | ||||||
|  | def test_on_plan_ended(client, thread, catch_event, compare): | ||||||
|  |     with catch_event("onPlanEnded") as x: | ||||||
|  |         client.createPlan(Plan(int(time()) + 120, "Wait for ending")) | ||||||
|  |         x.wait(180) | ||||||
|  |     assert subset( | ||||||
|  |         x.res, | ||||||
|  |         thread_id=client.uid if thread["type"] == ThreadType.USER else thread["id"], | ||||||
|  |         thread_type=thread["type"], | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # createPlan(self, plan, thread_id=None) | ||||||
|  | # editPlan(self, plan, new_plan) | ||||||
|  | # deletePlan(self, plan) | ||||||
|  | # changePlanParticipation(self, plan, take_part=True) | ||||||
|  |  | ||||||
|  | # onPlanCreated(self, mid=None, plan=None, author_id=None, thread_id=None, thread_type=None, ts=None, metadata=None, msg=None) | ||||||
|  | # onPlanEnded(self, mid=None, plan=None, thread_id=None, thread_type=None, ts=None, metadata=None, msg=None) | ||||||
|  | # onPlanEdited(self, mid=None, plan=None, author_id=None, thread_id=None, thread_type=None, ts=None, metadata=None, msg=None) | ||||||
|  | # onPlanDeleted(self, mid=None, plan=None, author_id=None, thread_id=None, thread_type=None, ts=None, metadata=None, msg=None) | ||||||
|  | # onPlanParticipation(self, mid=None, plan=None, take_part=None, author_id=None, thread_id=None, thread_type=None, ts=None, metadata=None, msg=None) | ||||||
|  |  | ||||||
|  | # fetchPlanInfo(self, plan_id) | ||||||
							
								
								
									
										107
									
								
								tests/test_polls.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,107 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from fbchat.models import Poll, PollOption, ThreadType | ||||||
|  | from utils import random_hex, subset | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture( | ||||||
|  |     scope="module", | ||||||
|  |     params=[ | ||||||
|  |         Poll(title=random_hex(), options=[]), | ||||||
|  |         Poll( | ||||||
|  |             title=random_hex(), | ||||||
|  |             options=[ | ||||||
|  |                 PollOption(random_hex(), vote=True), | ||||||
|  |                 PollOption(random_hex(), vote=True), | ||||||
|  |             ], | ||||||
|  |         ), | ||||||
|  |         Poll( | ||||||
|  |             title=random_hex(), | ||||||
|  |             options=[ | ||||||
|  |                 PollOption(random_hex(), vote=False), | ||||||
|  |                 PollOption(random_hex(), vote=False), | ||||||
|  |             ], | ||||||
|  |         ), | ||||||
|  |         Poll( | ||||||
|  |             title=random_hex(), | ||||||
|  |             options=[ | ||||||
|  |                 PollOption(random_hex(), vote=True), | ||||||
|  |                 PollOption(random_hex(), vote=True), | ||||||
|  |                 PollOption(random_hex(), vote=False), | ||||||
|  |                 PollOption(random_hex(), vote=False), | ||||||
|  |                 PollOption(random_hex()), | ||||||
|  |                 PollOption(random_hex()), | ||||||
|  |             ], | ||||||
|  |         ), | ||||||
|  |         pytest.param( | ||||||
|  |             Poll(title=None, options=[]), marks=[pytest.mark.xfail(raises=ValueError)] | ||||||
|  |         ), | ||||||
|  |     ], | ||||||
|  | ) | ||||||
|  | def poll_data(request, client1, group, catch_event): | ||||||
|  |     with catch_event("onPollCreated") as x: | ||||||
|  |         client1.createPoll(request.param, thread_id=group["id"]) | ||||||
|  |     options = client1.fetchPollOptions(x.res["poll"].uid) | ||||||
|  |     return x.res, request.param, options | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_create_poll(client1, group, catch_event, poll_data): | ||||||
|  |     event, poll, _ = poll_data | ||||||
|  |     assert subset( | ||||||
|  |         event, | ||||||
|  |         author_id=client1.uid, | ||||||
|  |         thread_id=group["id"], | ||||||
|  |         thread_type=ThreadType.GROUP, | ||||||
|  |     ) | ||||||
|  |     assert subset( | ||||||
|  |         vars(event["poll"]), title=poll.title, options_count=len(poll.options) | ||||||
|  |     ) | ||||||
|  |     for recv_option in event[ | ||||||
|  |         "poll" | ||||||
|  |     ].options:  # The recieved options may not be the full list | ||||||
|  |         (old_option,) = list(filter(lambda o: o.text == recv_option.text, poll.options)) | ||||||
|  |         voters = [client1.uid] if old_option.vote else [] | ||||||
|  |         assert subset( | ||||||
|  |             vars(recv_option), voters=voters, votes_count=len(voters), vote=False | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_fetch_poll_options(client1, group, catch_event, poll_data): | ||||||
|  |     _, poll, options = poll_data | ||||||
|  |     assert len(options) == len(poll.options) | ||||||
|  |     for option in options: | ||||||
|  |         assert subset(vars(option)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.trylast | ||||||
|  | def test_update_poll_vote(client1, group, catch_event, poll_data): | ||||||
|  |     event, poll, options = poll_data | ||||||
|  |     new_vote_ids = [o.uid for o in options[0 : len(options) : 2] if not o.vote] | ||||||
|  |     re_vote_ids = [o.uid for o in options[0 : len(options) : 2] if o.vote] | ||||||
|  |     new_options = [random_hex(), random_hex()] | ||||||
|  |     with catch_event("onPollVoted") as x: | ||||||
|  |         client1.updatePollVote( | ||||||
|  |             event["poll"].uid, | ||||||
|  |             option_ids=new_vote_ids + re_vote_ids, | ||||||
|  |             new_options=new_options, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     assert subset( | ||||||
|  |         x.res, | ||||||
|  |         author_id=client1.uid, | ||||||
|  |         thread_id=group["id"], | ||||||
|  |         thread_type=ThreadType.GROUP, | ||||||
|  |     ) | ||||||
|  |     assert subset( | ||||||
|  |         vars(x.res["poll"]), title=poll.title, options_count=len(options + new_options) | ||||||
|  |     ) | ||||||
|  |     for o in new_vote_ids: | ||||||
|  |         assert o in x.res["added_options"] | ||||||
|  |     assert len(x.res["added_options"]) == len(new_vote_ids) + len(new_options) | ||||||
|  |     assert set(x.res["removed_options"]) == set( | ||||||
|  |         o.uid for o in options if o.vote and o.uid not in re_vote_ids | ||||||
|  |     ) | ||||||
							
								
								
									
										18
									
								
								tests/test_search.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | from fbchat.models import ThreadType | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_search_for(client1): | ||||||
|  |     users = client1.searchForUsers("Mark Zuckerberg") | ||||||
|  |     assert len(users) > 0 | ||||||
|  |  | ||||||
|  |     u = users[0] | ||||||
|  |  | ||||||
|  |     assert u.uid == "4" | ||||||
|  |     assert u.type == ThreadType.USER | ||||||
|  |     assert u.photo[:4] == "http" | ||||||
|  |     assert u.url[:4] == "http" | ||||||
|  |     assert u.name == "Mark Zuckerberg" | ||||||
							
								
								
									
										125
									
								
								tests/test_send.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,125 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from os import path | ||||||
|  | from fbchat.models import FBchatFacebookError, Message, Mention | ||||||
|  | from utils import subset, STICKER_LIST, EMOJI_LIST, TEXT_LIST | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("text", TEXT_LIST) | ||||||
|  | def test_send_text(client, catch_event, compare, text): | ||||||
|  |     with catch_event("onMessage") as x: | ||||||
|  |         mid = client.sendMessage(text) | ||||||
|  |  | ||||||
|  |     assert compare(x, mid=mid, message=text) | ||||||
|  |     assert subset(vars(x.res["message_object"]), uid=mid, author=client.uid, text=text) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("emoji, emoji_size", EMOJI_LIST) | ||||||
|  | def test_send_emoji(client, catch_event, compare, emoji, emoji_size): | ||||||
|  |     with catch_event("onMessage") as x: | ||||||
|  |         mid = client.sendEmoji(emoji, emoji_size) | ||||||
|  |  | ||||||
|  |     assert compare(x, mid=mid, message=emoji) | ||||||
|  |     assert subset( | ||||||
|  |         vars(x.res["message_object"]), | ||||||
|  |         uid=mid, | ||||||
|  |         author=client.uid, | ||||||
|  |         text=emoji, | ||||||
|  |         emoji_size=emoji_size, | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_send_mentions(client, catch_event, compare, message_with_mentions): | ||||||
|  |     with catch_event("onMessage") as x: | ||||||
|  |         mid = client.send(message_with_mentions) | ||||||
|  |  | ||||||
|  |     assert compare(x, mid=mid, message=message_with_mentions.text) | ||||||
|  |     assert subset( | ||||||
|  |         vars(x.res["message_object"]), | ||||||
|  |         uid=mid, | ||||||
|  |         author=client.uid, | ||||||
|  |         text=message_with_mentions.text, | ||||||
|  |     ) | ||||||
|  |     # The mentions are not ordered by offset | ||||||
|  |     for m in x.res["message_object"].mentions: | ||||||
|  |         assert vars(m) in [vars(x) for x in message_with_mentions.mentions] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("sticker", STICKER_LIST) | ||||||
|  | def test_send_sticker(client, catch_event, compare, sticker): | ||||||
|  |     with catch_event("onMessage") as x: | ||||||
|  |         mid = client.send(Message(sticker=sticker)) | ||||||
|  |  | ||||||
|  |     assert compare(x, mid=mid) | ||||||
|  |     assert subset(vars(x.res["message_object"]), uid=mid, author=client.uid) | ||||||
|  |     assert subset(vars(x.res["message_object"].sticker), uid=sticker.uid) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Kept for backwards compatibility | ||||||
|  | @pytest.mark.parametrize( | ||||||
|  |     "method_name, url", | ||||||
|  |     [ | ||||||
|  |         ( | ||||||
|  |             "sendRemoteImage", | ||||||
|  |             "https://github.com/carpedm20/fbchat/raw/master/tests/image.png", | ||||||
|  |         ), | ||||||
|  |         ("sendLocalImage", path.join(path.dirname(__file__), "resources", "image.png")), | ||||||
|  |     ], | ||||||
|  | ) | ||||||
|  | def test_send_images(client, catch_event, compare, method_name, url): | ||||||
|  |     text = "An image sent with {}".format(method_name) | ||||||
|  |     with catch_event("onMessage") as x: | ||||||
|  |         mid = getattr(client, method_name)(url, Message(text)) | ||||||
|  |  | ||||||
|  |     assert compare(x, mid=mid, message=text) | ||||||
|  |     assert subset(vars(x.res["message_object"]), uid=mid, author=client.uid, text=text) | ||||||
|  |     assert x.res["message_object"].attachments[0] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_send_local_files(client, catch_event, compare): | ||||||
|  |     files = [ | ||||||
|  |         "image.png", | ||||||
|  |         "image.jpg", | ||||||
|  |         "image.gif", | ||||||
|  |         "file.json", | ||||||
|  |         "file.txt", | ||||||
|  |         "audio.mp3", | ||||||
|  |         "video.mp4", | ||||||
|  |     ] | ||||||
|  |     text = "Files sent locally" | ||||||
|  |     with catch_event("onMessage") as x: | ||||||
|  |         mid = client.sendLocalFiles( | ||||||
|  |             [path.join(path.dirname(__file__), "resources", f) for f in files], | ||||||
|  |             message=Message(text), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     assert compare(x, mid=mid, message=text) | ||||||
|  |     assert subset(vars(x.res["message_object"]), uid=mid, author=client.uid, text=text) | ||||||
|  |     assert len(x.res["message_object"].attachments) == len(files) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # To be changed when merged into master | ||||||
|  | def test_send_remote_files(client, catch_event, compare): | ||||||
|  |     files = ["image.png", "data.json"] | ||||||
|  |     text = "Files sent from remote" | ||||||
|  |     with catch_event("onMessage") as x: | ||||||
|  |         mid = client.sendRemoteFiles( | ||||||
|  |             [ | ||||||
|  |                 "https://github.com/carpedm20/fbchat/raw/master/tests/{}".format(f) | ||||||
|  |                 for f in files | ||||||
|  |             ], | ||||||
|  |             message=Message(text), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     assert compare(x, mid=mid, message=text) | ||||||
|  |     assert subset(vars(x.res["message_object"]), uid=mid, author=client.uid, text=text) | ||||||
|  |     assert len(x.res["message_object"].attachments) == len(files) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("wave_first", [True, False]) | ||||||
|  | def test_wave(client, wave_first): | ||||||
|  |     client.wave(wave_first) | ||||||
							
								
								
									
										12
									
								
								tests/test_tests.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_catch_event(client2, catch_event): | ||||||
|  |     mid = "test" | ||||||
|  |     with catch_event("onMessage") as x: | ||||||
|  |         client2.onMessage(mid=mid) | ||||||
|  |     assert x.res["mid"] == mid | ||||||
							
								
								
									
										164
									
								
								tests/test_thread_interraction.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,164 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from fbchat.models import ( | ||||||
|  |     Message, | ||||||
|  |     ThreadType, | ||||||
|  |     FBchatFacebookError, | ||||||
|  |     TypingStatus, | ||||||
|  |     ThreadColor, | ||||||
|  | ) | ||||||
|  | from utils import random_hex, subset | ||||||
|  | from os import path | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_remove_from_and_add_to_group(client1, client2, group, catch_event): | ||||||
|  |     # Test both methods, while ensuring that the user gets added to the group | ||||||
|  |     try: | ||||||
|  |         with catch_event("onPersonRemoved") as x: | ||||||
|  |             client1.removeUserFromGroup(client2.uid, group["id"]) | ||||||
|  |         assert subset( | ||||||
|  |             x.res, removed_id=client2.uid, author_id=client1.uid, thread_id=group["id"] | ||||||
|  |         ) | ||||||
|  |     finally: | ||||||
|  |         with catch_event("onPeopleAdded") as x: | ||||||
|  |             client1.addUsersToGroup(client2.uid, group["id"]) | ||||||
|  |         assert subset( | ||||||
|  |             x.res, added_ids=[client2.uid], author_id=client1.uid, thread_id=group["id"] | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_remove_from_and_add_admins_to_group(client1, client2, group, catch_event): | ||||||
|  |     # Test both methods, while ensuring that the user gets added as group admin | ||||||
|  |     try: | ||||||
|  |         with catch_event("onAdminRemoved") as x: | ||||||
|  |             client1.removeGroupAdmins(client2.uid, group["id"]) | ||||||
|  |         assert subset( | ||||||
|  |             x.res, removed_id=client2.uid, author_id=client1.uid, thread_id=group["id"] | ||||||
|  |         ) | ||||||
|  |     finally: | ||||||
|  |         with catch_event("onAdminAdded") as x: | ||||||
|  |             client1.addGroupAdmins(client2.uid, group["id"]) | ||||||
|  |         assert subset( | ||||||
|  |             x.res, added_id=client2.uid, author_id=client1.uid, thread_id=group["id"] | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_change_title(client1, group, catch_event): | ||||||
|  |     title = random_hex() | ||||||
|  |     with catch_event("onTitleChange") as x: | ||||||
|  |         client1.changeThreadTitle(title, group["id"], thread_type=ThreadType.GROUP) | ||||||
|  |     assert subset( | ||||||
|  |         x.res, | ||||||
|  |         author_id=client1.uid, | ||||||
|  |         new_title=title, | ||||||
|  |         thread_id=group["id"], | ||||||
|  |         thread_type=ThreadType.GROUP, | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_change_nickname(client, client_all, catch_event, compare): | ||||||
|  |     nickname = random_hex() | ||||||
|  |     with catch_event("onNicknameChange") as x: | ||||||
|  |         client.changeNickname(nickname, client_all.uid) | ||||||
|  |     assert compare(x, changed_for=client_all.uid, new_nickname=nickname) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize( | ||||||
|  |     "emoji", | ||||||
|  |     [ | ||||||
|  |         "😀", | ||||||
|  |         "😂", | ||||||
|  |         "😕", | ||||||
|  |         "😍", | ||||||
|  |         pytest.param("🙃", marks=[pytest.mark.xfail(raises=FBchatFacebookError)]), | ||||||
|  |         pytest.param( | ||||||
|  |             "not an emoji", marks=[pytest.mark.xfail(raises=FBchatFacebookError)] | ||||||
|  |         ), | ||||||
|  |     ], | ||||||
|  | ) | ||||||
|  | def test_change_emoji(client, catch_event, compare, emoji): | ||||||
|  |     with catch_event("onEmojiChange") as x: | ||||||
|  |         client.changeThreadEmoji(emoji) | ||||||
|  |     assert compare(x, new_emoji=emoji) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_change_image_local(client1, group, catch_event): | ||||||
|  |     url = path.join(path.dirname(__file__), "resources", "image.png") | ||||||
|  |     with catch_event("onImageChange") as x: | ||||||
|  |         image_id = client1.changeGroupImageLocal(url, group["id"]) | ||||||
|  |     assert subset( | ||||||
|  |         x.res, new_image=image_id, author_id=client1.uid, thread_id=group["id"] | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # To be changed when merged into master | ||||||
|  | def test_change_image_remote(client1, group, catch_event): | ||||||
|  |     url = "https://github.com/carpedm20/fbchat/raw/master/tests/image.png" | ||||||
|  |     with catch_event("onImageChange") as x: | ||||||
|  |         image_id = client1.changeGroupImageRemote(url, group["id"]) | ||||||
|  |     assert subset( | ||||||
|  |         x.res, new_image=image_id, author_id=client1.uid, thread_id=group["id"] | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize( | ||||||
|  |     "color", | ||||||
|  |     [ | ||||||
|  |         x | ||||||
|  |         if x in [ThreadColor.MESSENGER_BLUE, ThreadColor.PUMPKIN] | ||||||
|  |         else pytest.param(x, marks=[pytest.mark.expensive()]) | ||||||
|  |         for x in ThreadColor | ||||||
|  |     ], | ||||||
|  | ) | ||||||
|  | def test_change_color(client, catch_event, compare, color): | ||||||
|  |     with catch_event("onColorChange") as x: | ||||||
|  |         client.changeThreadColor(color) | ||||||
|  |     assert compare(x, new_color=color) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail(raises=FBchatFacebookError, reason="Should fail, but doesn't") | ||||||
|  | def test_change_color_invalid(client): | ||||||
|  |     class InvalidColor: | ||||||
|  |         value = "#0077ff" | ||||||
|  |  | ||||||
|  |     client.changeThreadColor(InvalidColor()) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("status", TypingStatus) | ||||||
|  | def test_typing_status(client, catch_event, compare, status): | ||||||
|  |     with catch_event("onTyping") as x: | ||||||
|  |         client.setTypingStatus(status) | ||||||
|  |     assert compare(x, status=status) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("require_admin_approval", [True, False]) | ||||||
|  | def test_change_approval_mode(client1, group, catch_event, require_admin_approval): | ||||||
|  |     with catch_event("onApprovalModeChange") as x: | ||||||
|  |         client1.changeGroupApprovalMode(require_admin_approval, group["id"]) | ||||||
|  |  | ||||||
|  |     assert subset( | ||||||
|  |         x.res, | ||||||
|  |         approval_mode=require_admin_approval, | ||||||
|  |         author_id=client1.uid, | ||||||
|  |         thread_id=group["id"], | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("mute_time", [0, 10, 100, 1000, -1]) | ||||||
|  | def test_mute_thread(client, mute_time): | ||||||
|  |     assert client.muteThread(mute_time) | ||||||
|  |     assert client.unmuteThread() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_mute_thread_reactions(client): | ||||||
|  |     assert client.muteThreadReactions() | ||||||
|  |     assert client.unmuteThreadReactions() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_mute_thread_mentions(client): | ||||||
|  |     assert client.muteThreadMentions() | ||||||
|  |     assert client.unmuteThreadMentions() | ||||||
							
								
								
									
										114
									
								
								tests/utils.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,114 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import threading | ||||||
|  | import logging | ||||||
|  | import six | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from os import environ | ||||||
|  | from random import randrange | ||||||
|  | from contextlib import contextmanager | ||||||
|  | from six import viewitems | ||||||
|  | from fbchat import Client | ||||||
|  | from fbchat.models import ThreadType, EmojiSize, FBchatFacebookError, Sticker | ||||||
|  |  | ||||||
|  | log = logging.getLogger("fbchat.tests").addHandler(logging.NullHandler()) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | EMOJI_LIST = [ | ||||||
|  |     ("😆", EmojiSize.SMALL), | ||||||
|  |     ("😆", EmojiSize.MEDIUM), | ||||||
|  |     ("😆", EmojiSize.LARGE), | ||||||
|  |     # These fail in `catch_event` because the emoji is made into a sticker | ||||||
|  |     # This should be fixed | ||||||
|  |     pytest.param(None, EmojiSize.SMALL, marks=[pytest.mark.xfail()]), | ||||||
|  |     pytest.param(None, EmojiSize.MEDIUM, marks=[pytest.mark.xfail()]), | ||||||
|  |     pytest.param(None, EmojiSize.LARGE, marks=[pytest.mark.xfail()]), | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | STICKER_LIST = [ | ||||||
|  |     Sticker("767334476626295"), | ||||||
|  |     pytest.param(Sticker("0"), marks=[pytest.mark.xfail(raises=FBchatFacebookError)]), | ||||||
|  |     pytest.param(Sticker(None), marks=[pytest.mark.xfail(raises=FBchatFacebookError)]), | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | TEXT_LIST = [ | ||||||
|  |     "test_send", | ||||||
|  |     "😆", | ||||||
|  |     "\\\n\t%?&'\"", | ||||||
|  |     "ˁҭʚ¹Ʋջوװ՞ޱɣࠚԹБɑȑңКએ֭ʗыԈٌʼőԈ×௴nચϚࠖణٔє܅Ԇޑط", | ||||||
|  |     "a" * 20000,  # Maximum amount of characters you can send | ||||||
|  |     pytest.param("a" * 20001, marks=[pytest.mark.xfail(raises=FBchatFacebookError)]), | ||||||
|  |     pytest.param(None, marks=[pytest.mark.xfail(raises=FBchatFacebookError)]), | ||||||
|  | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ClientThread(threading.Thread): | ||||||
|  |     def __init__(self, client, *args, **kwargs): | ||||||
|  |         self.client = client | ||||||
|  |         self.should_stop = threading.Event() | ||||||
|  |         super(ClientThread, self).__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     def start(self): | ||||||
|  |         self.client.startListening() | ||||||
|  |         self.client.doOneListen()  # QPrimer, Facebook now knows we're about to start pulling | ||||||
|  |         super(ClientThread, self).start() | ||||||
|  |  | ||||||
|  |     def run(self): | ||||||
|  |         while not self.should_stop.is_set() and self.client.doOneListen(): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         self.client.stopListening() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if six.PY2: | ||||||
|  |     event_class = threading._Event | ||||||
|  | else: | ||||||
|  |     event_class = threading.Event | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CaughtValue(event_class): | ||||||
|  |     def set(self, res): | ||||||
|  |         self.res = res | ||||||
|  |         super(CaughtValue, self).set() | ||||||
|  |  | ||||||
|  |     def wait(self, timeout=3): | ||||||
|  |         super(CaughtValue, self).wait(timeout=timeout) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def random_hex(length=20): | ||||||
|  |     return "{:X}".format(randrange(16 ** length)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def subset(a, **b): | ||||||
|  |     print(a) | ||||||
|  |     print(b) | ||||||
|  |     return viewitems(b) <= viewitems(a) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def load_variable(name, cache): | ||||||
|  |     var = environ.get(name, None) | ||||||
|  |     if var is not None: | ||||||
|  |         if cache.get(name, None) != var: | ||||||
|  |             cache.set(name, var) | ||||||
|  |         return var | ||||||
|  |  | ||||||
|  |     var = cache.get(name, None) | ||||||
|  |     if var is None: | ||||||
|  |         raise ValueError("Variable {!r} neither in environment nor cache".format(name)) | ||||||
|  |     return var | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @contextmanager | ||||||
|  | def load_client(n, cache): | ||||||
|  |     client = Client( | ||||||
|  |         load_variable("client{}_email".format(n), cache), | ||||||
|  |         load_variable("client{}_password".format(n), cache), | ||||||
|  |         user_agent="Mozilla/5.0 (Windows NT 6.3; WOW64; ; NCT50_AAP285C84A1328) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/42.0.2311.90 Safari/537.36", | ||||||
|  |         session_cookies=cache.get("client{}_session".format(n), None), | ||||||
|  |         max_tries=1, | ||||||
|  |     ) | ||||||
|  |     yield client | ||||||
|  |     cache.set("client{}_session".format(n), client.getSession()) | ||||||