Compare commits
	
		
			246 Commits
		
	
	
		
			v1.6.1
			...
			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 | 
							
								
								
									
										7
									
								
								.bumpversion.cfg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								.bumpversion.cfg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| [bumpversion] | ||||
| current_version = 1.9.6 | ||||
| commit = True | ||||
| tag = True | ||||
|  | ||||
| [bumpversion:file:fbchat/__init__.py] | ||||
|  | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							| @@ -21,8 +21,8 @@ Traceback (most recent call last): | ||||
|   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) | ||||
|     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 | ||||
|   | ||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -35,3 +35,4 @@ tests.data | ||||
|  | ||||
| # Virtual environment | ||||
| venv/ | ||||
| .venv*/ | ||||
|   | ||||
							
								
								
									
										18
									
								
								.readthedocs.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										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 | ||||
							
								
								
									
										15
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -5,6 +5,7 @@ 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 | ||||
|  | ||||
| @@ -23,8 +24,13 @@ jobs: | ||||
|     sudo: required | ||||
|   - python: pypy3.5 | ||||
|  | ||||
|   - name: Lint | ||||
|     before_install: skip | ||||
|     install: pip install black | ||||
|     script: black --check --verbose . | ||||
|  | ||||
|   - stage: deploy | ||||
|     name: Github Releases | ||||
|     name: GitHub Releases | ||||
|     if: tag IS present | ||||
|     install: skip | ||||
|     script: flit build | ||||
| @@ -34,7 +40,7 @@ jobs: | ||||
|       file_glob: true | ||||
|       file: dist/* | ||||
|       skip_cleanup: true | ||||
|       draft: true | ||||
|       draft: false | ||||
|       on: | ||||
|         tags: true | ||||
|  | ||||
| @@ -48,3 +54,8 @@ jobs: | ||||
|       script: flit publish | ||||
|       on: | ||||
|         tags: true | ||||
|  | ||||
| notifications: | ||||
|   email: | ||||
|     on_success: never | ||||
|     on_failure: change | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| Contributing to fbchat | ||||
| ====================== | ||||
| Contributing to ``fbchat`` | ||||
| ========================== | ||||
|  | ||||
| Thanks for reading this, all contributions are very much welcome! | ||||
|  | ||||
| @@ -9,6 +9,22 @@ That means that if you're submitting a breaking change, it will probably take a | ||||
| 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 | ||||
| ------------------- | ||||
|  | ||||
|   | ||||
							
								
								
									
										16
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.rst
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | ||||
| fbchat: Facebook Chat (Messenger) for Python | ||||
| ============================================ | ||||
| ``fbchat``: Facebook Chat (Messenger) for Python | ||||
| ================================================ | ||||
|  | ||||
| .. image:: https://img.shields.io/badge/license-BSD-blue.svg | ||||
|     :target: https://github.com/carpedm20/fbchat/tree/master/LICENSE | ||||
| @@ -9,7 +9,7 @@ fbchat: Facebook Chat (Messenger) for Python | ||||
|     :target: https://pypi.python.org/pypi/fbchat | ||||
|     :alt: Supported python versions: 2.7, 3.4, 3.5, 3.6, 3.7 and pypy | ||||
|  | ||||
| .. image:: https://readthedocs.org/projects/fbchat/badge/?version=master | ||||
| .. image:: https://readthedocs.org/projects/fbchat/badge/?version=latest | ||||
|     :target: https://fbchat.readthedocs.io | ||||
|     :alt: Documentation | ||||
|  | ||||
| @@ -17,6 +17,10 @@ fbchat: Facebook Chat (Messenger) for Python | ||||
|     :target: https://travis-ci.org/carpedm20/fbchat | ||||
|     :alt: Travis CI | ||||
|  | ||||
| .. image:: https://img.shields.io/badge/code%20style-black-000000.svg | ||||
|     :target: https://github.com/ambv/black | ||||
|     :alt: Code style | ||||
|  | ||||
| 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>`__. | ||||
|  | ||||
| @@ -31,14 +35,12 @@ Installation: | ||||
|  | ||||
|     $ pip install fbchat | ||||
|  | ||||
| You can also install from source, by using `flit`: | ||||
| You can also install from source if you have ``pip>=19.0``: | ||||
|  | ||||
| .. code-block:: | ||||
|  | ||||
|     $ pip install flit | ||||
|     $ git clone https://github.com/carpedm20/fbchat.git | ||||
|     $ cd fbchat | ||||
|     $ flit install | ||||
|     $ pip install fbchat | ||||
|  | ||||
|  | ||||
| Maintainer | ||||
|   | ||||
| @@ -3,8 +3,7 @@ | ||||
|  | ||||
| # You can set these variables from the command line. | ||||
| SPHINXOPTS    = | ||||
| SPHINXBUILD   = python3.6 -msphinx | ||||
| SPHINXPROJ    = fbchat | ||||
| SPHINXBUILD   = sphinx-build | ||||
| SOURCEDIR     = . | ||||
| BUILDDIR      = _build | ||||
|  | ||||
|   | ||||
							
								
								
									
										81
									
								
								docs/api.rst
									
									
									
									
									
								
							
							
						
						
									
										81
									
								
								docs/api.rst
									
									
									
									
									
								
							| @@ -1,44 +1,79 @@ | ||||
| .. module:: fbchat | ||||
| .. highlight:: python | ||||
| .. _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. | ||||
|  | ||||
|  | ||||
| .. _api_client: | ||||
|  | ||||
| Client | ||||
| ------ | ||||
|  | ||||
| This is the main class of `fbchat`, which contains all the methods you use to interact with Facebook. | ||||
| You can extend this class, and overwrite the events, to provide custom event handling (mainly used while listening) | ||||
| .. autoclass:: Client | ||||
|  | ||||
| .. autoclass:: Client(email, password, user_agent=None, max_tries=5, session_cookies=None, logging_level=logging.INFO) | ||||
|     :members: | ||||
| Threads | ||||
| ------- | ||||
|  | ||||
| .. autoclass:: Thread() | ||||
| .. autoclass:: ThreadType(Enum) | ||||
|     :undoc-members: | ||||
| .. autoclass:: Page() | ||||
| .. autoclass:: User() | ||||
| .. autoclass:: Group() | ||||
|  | ||||
| .. _api_models: | ||||
| Messages | ||||
| -------- | ||||
|  | ||||
| Models | ||||
| ------ | ||||
|  | ||||
| These models are used in various functions, both as inputs and return values. | ||||
| A good tip is to write ``from fbchat.models import *`` at the start of your source, so you can use these models freely | ||||
|  | ||||
| .. automodule:: fbchat.models | ||||
|     :members: | ||||
| .. autoclass:: Message | ||||
| .. autoclass:: Mention | ||||
| .. autoclass:: EmojiSize(Enum) | ||||
|     :undoc-members: | ||||
| .. autoclass:: MessageReaction(Enum) | ||||
|     :undoc-members: | ||||
|  | ||||
| Exceptions | ||||
| ---------- | ||||
|  | ||||
| .. _api_utils: | ||||
| .. autoexception:: FBchatException() | ||||
| .. autoexception:: FBchatFacebookError() | ||||
| .. autoexception:: FBchatUserError() | ||||
|  | ||||
| Utils | ||||
| ----- | ||||
| Attachments | ||||
| ----------- | ||||
|  | ||||
| These functions and values are used internally by fbchat, and are subject to change. Do **NOT** rely on these to be backwards compatible! | ||||
| .. autoclass:: Attachment() | ||||
| .. autoclass:: ShareAttachment() | ||||
| .. autoclass:: Sticker() | ||||
| .. autoclass:: LocationAttachment() | ||||
| .. autoclass:: LiveLocationAttachment() | ||||
| .. autoclass:: FileAttachment() | ||||
| .. autoclass:: AudioAttachment() | ||||
| .. autoclass:: ImageAttachment() | ||||
| .. autoclass:: VideoAttachment() | ||||
| .. autoclass:: ImageAttachment() | ||||
|  | ||||
| .. automodule:: fbchat.utils | ||||
|     :members: | ||||
| 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: | ||||
|   | ||||
							
								
								
									
										249
									
								
								docs/conf.py
									
									
									
									
									
								
							
							
						
						
									
										249
									
								
								docs/conf.py
									
									
									
									
									
								
							| @@ -1,191 +1,208 @@ | ||||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| # | ||||
| # fbchat documentation build configuration file, created by | ||||
| # sphinx-quickstart on Thu May 25 15:43:01 2017. | ||||
| # Configuration file for the Sphinx documentation builder. | ||||
| # | ||||
| # This file is execfile()d with the current directory set to its | ||||
| # containing dir. | ||||
| # | ||||
| # Note that not all possible configuration values are present in this | ||||
| # autogenerated file. | ||||
| # | ||||
| # All configuration values have a default; values that are commented out | ||||
| # serve to show the default. | ||||
| # 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 | ||||
|  | ||||
| # If extensions (or modules to document with autodoc) are in another directory, | ||||
| # add these directories to sys.path here. If the directory is relative to the | ||||
| # documentation root, use os.path.abspath to make it absolute, like shown here. | ||||
| # -- Path setup -------------------------------------------------------------- | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| sys.path.insert(0, os.path.abspath('..')) | ||||
|  | ||||
| sys.path.insert(0, os.path.abspath("..")) | ||||
|  | ||||
| import fbchat | ||||
| import tests | ||||
| from fbchat import __copyright__, __author__, __version__, __description__ | ||||
|  | ||||
| # -- 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 ------------------------------------------------ | ||||
| # -- General configuration --------------------------------------------------- | ||||
|  | ||||
| # If your documentation needs a minimal Sphinx version, state it here. | ||||
| # | ||||
| # needs_sphinx = '1.0' | ||||
| 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.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 suffix(es) of source filenames. | ||||
| # You can specify multiple suffix as a list of string: | ||||
| # | ||||
| # source_suffix = ['.rst', '.md'] | ||||
| source_suffix = '.rst' | ||||
| templates_path = ["_templates"] | ||||
|  | ||||
| # The master toctree document. | ||||
| master_doc = 'index' | ||||
|  | ||||
| # General information about the project. | ||||
| project = 'fbchat' | ||||
| title = 'fbchat Documentation' | ||||
| copyright = __copyright__ | ||||
| author = __author__ | ||||
| description = __description__ | ||||
|  | ||||
| # The version info for the project you're documenting, acts as replacement for | ||||
| # |version| and |release|, also used in various other places throughout the | ||||
| # built documents. | ||||
| # | ||||
| # The short X.Y version. | ||||
| version = __version__ | ||||
| # The full version, including alpha/beta/rc tags. | ||||
| release = __version__ | ||||
|  | ||||
| # The language for content autogenerated by Sphinx. Refer to documentation | ||||
| # for a list of supported languages. | ||||
| # | ||||
| # This is also used if you do content translation via gettext catalogs. | ||||
| # Usually you set "language" from the command line for these cases. | ||||
| language = None | ||||
| master_doc = "index" | ||||
|  | ||||
| # List of patterns, relative to source directory, that match files and | ||||
| # directories to ignore when looking for source files. | ||||
| # This patterns also effect to html_static_path and html_extra_path | ||||
| exclude_patterns = ['_build', 'Thumbs.db', '.DS_Store'] | ||||
| # This pattern also affects html_static_path and html_extra_path. | ||||
| exclude_patterns = ["_build", "Thumbs.db", ".DS_Store"] | ||||
|  | ||||
| # The name of the Pygments (syntax highlighting) style to use. | ||||
| pygments_style = 'sphinx' | ||||
| rst_prolog = ".. currentmodule:: " + project | ||||
|  | ||||
| # If true, `todo` and `todoList` produce output, else they produce nothing. | ||||
| todo_include_todos = True | ||||
| # 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 ---------------------------------------------- | ||||
| # -- 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' | ||||
| 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 = {} | ||||
| html_theme_options = { | ||||
|     "show_powered_by": False, | ||||
|     "github_user": "carpedm20", | ||||
|     "github_repo": project, | ||||
|     "github_banner": True, | ||||
|     "show_related": False, | ||||
| } | ||||
|  | ||||
| # Add any paths that contain custom static files (such as style sheets) here, | ||||
| # relative to this directory. They are copied after the builtin static files, | ||||
| # so a file named "default.css" will overwrite the builtin "default.css". | ||||
| html_static_path = ['_static'] | ||||
| # 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 ------------------------------------------ | ||||
| # -- Options for HTMLHelp output --------------------------------------------- | ||||
|  | ||||
| # Output file base name for HTML help builder. | ||||
| htmlhelp_basename = project + 'doc' | ||||
| htmlhelp_basename = project + "doc" | ||||
|  | ||||
|  | ||||
| # -- Options for LaTeX output --------------------------------------------- | ||||
|  | ||||
| latex_elements = { | ||||
|     # The paper size ('letterpaper' or 'a4paper'). | ||||
|     # | ||||
|     # 'papersize': 'letterpaper', | ||||
|  | ||||
|     # The font size ('10pt', '11pt' or '12pt'). | ||||
|     # | ||||
|     # 'pointsize': '10pt', | ||||
|  | ||||
|     # Additional stuff for the LaTeX preamble. | ||||
|     # | ||||
|     # 'preamble': '', | ||||
|  | ||||
|     # Latex figure (float) alignment | ||||
|     # | ||||
|     # 'figure_align': 'htbp', | ||||
| } | ||||
| # -- 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', title, | ||||
|      author, 'manual'), | ||||
| ] | ||||
| latex_documents = [(master_doc, project + ".tex", fbchat.__title__, author, "manual")] | ||||
|  | ||||
|  | ||||
| # -- Options for manual page output --------------------------------------- | ||||
| # -- 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, title, | ||||
|      [author], 1) | ||||
|     (master_doc, project, fbchat.__title__, [x.strip() for x in author.split(";")], 1) | ||||
| ] | ||||
|  | ||||
|  | ||||
| # -- Options for Texinfo output ------------------------------------------- | ||||
| # -- 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, title, | ||||
|      author, project, description, | ||||
|      'Miscellaneous'), | ||||
|     ( | ||||
|         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/3/': None} | ||||
| 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 | ||||
|  | ||||
| add_function_parentheses = False | ||||
| todo_link_only = True | ||||
|  | ||||
| html_theme_options = { | ||||
|     'show_powered_by': False, | ||||
|     'github_user': 'carpedm20', | ||||
|     'github_repo': project, | ||||
|     'github_banner': True, | ||||
|     'show_related': False | ||||
| } | ||||
| # -- Options for napoleon extension ---------------------------------------------- | ||||
|  | ||||
| html_sidebars = { | ||||
|     '**': ['sidebar.html', 'searchbox.html'] | ||||
| } | ||||
| # Use Google style docstrings | ||||
| napoleon_google_docstring = True | ||||
| napoleon_numpy_docstring = False | ||||
|  | ||||
| html_show_sphinx = False | ||||
| html_show_sourcelink = False | ||||
| autoclass_content = 'init' | ||||
| html_short_title = description | ||||
| # 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 | ||||
|   | ||||
| @@ -1,16 +1,15 @@ | ||||
| .. highlight:: python | ||||
| .. _examples: | ||||
|  | ||||
| Examples | ||||
| ======== | ||||
|  | ||||
| These are a few examples on how to use `fbchat`. Remember to swap out `<email>` and `<password>` for your email and password | ||||
| 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` | ||||
| This will show basic usage of ``fbchat`` | ||||
|  | ||||
| .. literalinclude:: ../examples/basic_usage.py | ||||
|  | ||||
| @@ -18,7 +17,7 @@ This will show basic usage of `fbchat` | ||||
| Interacting with Threads | ||||
| ------------------------ | ||||
|  | ||||
| This will interact with the thread in every way `fbchat` supports | ||||
| This will interact with the thread in every way ``fbchat`` supports | ||||
|  | ||||
| .. literalinclude:: ../examples/interract.py | ||||
|  | ||||
| @@ -31,8 +30,8 @@ This will show the different ways of fetching information about users and thread | ||||
| .. literalinclude:: ../examples/fetch.py | ||||
|  | ||||
|  | ||||
| Echobot | ||||
| ------- | ||||
| ``Echobot`` | ||||
| ----------- | ||||
|  | ||||
| This will reply to any message with the same message | ||||
|  | ||||
| @@ -42,7 +41,7 @@ This will reply to any message with the same message | ||||
| Remove Bot | ||||
| ---------- | ||||
|  | ||||
| This will remove a user from a group if they write the message `Remove me!` | ||||
| This will remove a user from a group if they write the message ``Remove me!`` | ||||
|  | ||||
| .. literalinclude:: ../examples/removebot.py | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,3 @@ | ||||
| .. highlight:: python | ||||
| .. module:: fbchat | ||||
| .. _faq: | ||||
|  | ||||
| FAQ | ||||
| @@ -11,7 +9,7 @@ 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 | ||||
| Downgrade to an earlier version of ``fbchat``, run this command | ||||
|  | ||||
| .. code-block:: sh | ||||
|  | ||||
| @@ -23,14 +21,14 @@ 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 ;) | ||||
| 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>`_. | ||||
| 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 | ||||
|   | ||||
| @@ -1,5 +1,3 @@ | ||||
| .. highlight:: python | ||||
| .. module:: fbchat | ||||
| .. 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 | ||||
| @@ -8,8 +6,8 @@ | ||||
| .. 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 | ||||
| ============================================ | ||||
| ``fbchat``: Facebook Chat (Messenger) for Python | ||||
| ================================================ | ||||
|  | ||||
| Release v\ |version|. (:ref:`install`) | ||||
|  | ||||
| @@ -30,14 +28,14 @@ This project was inspired by `facebook-chat-api <https://github.com/Schmavery/fa | ||||
|  | ||||
| **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: | ||||
| Currently ``fbchat`` support Python 2.7, 3.4, 3.5 and 3.6: | ||||
|  | ||||
| `fbchat` works by emulating the browser. | ||||
| ``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 | ||||
|     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, | ||||
| @@ -46,9 +44,9 @@ Therefore, this API requires the credentials of a Facebook account. | ||||
|  | ||||
| .. 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. | ||||
|     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` | ||||
| If you're already familiar with the basics of how Facebook works internally, go to :ref:`examples` to see example usage of ``fbchat`` | ||||
|  | ||||
|  | ||||
| Overview | ||||
|   | ||||
| @@ -1,13 +1,14 @@ | ||||
| .. highlight:: sh | ||||
| .. _install: | ||||
|  | ||||
| Installation | ||||
| ============ | ||||
|  | ||||
| Pip Install fbchat | ||||
| ------------------ | ||||
| Install using pip | ||||
| ----------------- | ||||
|  | ||||
| To install fbchat, run this command:: | ||||
| To install ``fbchat``, run this command: | ||||
|  | ||||
| .. code-block:: sh | ||||
|  | ||||
|     $ pip install fbchat | ||||
|  | ||||
| @@ -18,19 +19,25 @@ can guide you through the process. | ||||
| Get the Source Code | ||||
| ------------------- | ||||
|  | ||||
| fbchat is developed on GitHub, where the code is | ||||
| ``fbchat`` is developed on GitHub, where the code is | ||||
| `always available <https://github.com/carpedm20/fbchat>`_. | ||||
|  | ||||
| You can either clone the public repository:: | ||||
| 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>`_:: | ||||
| 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:: | ||||
| package, or install it into your site-packages easily: | ||||
|  | ||||
| .. code-block:: sh | ||||
|  | ||||
|     $ python setup.py install | ||||
|   | ||||
| @@ -1,11 +1,9 @@ | ||||
| .. highlight:: python | ||||
| .. module:: fbchat | ||||
| .. _intro: | ||||
|  | ||||
| Introduction | ||||
| ============ | ||||
|  | ||||
| `fbchat` uses your email and password to communicate with the Facebook server. | ||||
| ``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 | ||||
|  | ||||
| @@ -26,9 +24,9 @@ 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 neccesary here, later code snippets will assume you've done this | ||||
|     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`) | ||||
| 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`:: | ||||
| @@ -48,9 +46,9 @@ Threads | ||||
|  | ||||
| A thread can refer to two things: A Messenger group chat or a single Facebook user | ||||
|  | ||||
| :class:`models.ThreadType` is an enumerator with two values: ``USER`` and ``GROUP``. | ||||
| :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 | ||||
| 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` | ||||
| @@ -87,7 +85,7 @@ 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`, | ||||
| 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:: | ||||
|  | ||||
| @@ -100,17 +98,17 @@ This snippet shows how to send a message, and then use the returned ID to react | ||||
| Interacting with Threads | ||||
| ------------------------ | ||||
|  | ||||
| `fbchat` provides multiple functions for 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:: | ||||
| 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` | ||||
| You can see a full example showing all the possible thread interactions with ``fbchat`` by going to :ref:`examples` | ||||
|  | ||||
|  | ||||
| .. _intro_fetching: | ||||
| @@ -118,7 +116,7 @@ You can see a full example showing all the possible thread interactions with `fb | ||||
| Fetching Information | ||||
| -------------------- | ||||
|  | ||||
| You can use `fbchat` to fetch basic information like user names, profile pictures, thread names and user IDs | ||||
| 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:: | ||||
| @@ -127,12 +125,12 @@ The following snippet will search for users by their name, take the first (and m | ||||
|     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)) | ||||
|     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` | ||||
| You can see a full example showing all the possible ways to fetch information with ``fbchat`` by going to :ref:`examples` | ||||
|  | ||||
|  | ||||
| .. _intro_sessions: | ||||
| @@ -140,7 +138,7 @@ You can see a full example showing all the possible ways to fetch information wi | ||||
| Sessions | ||||
| -------- | ||||
|  | ||||
| `fbchat` provides functions to retrieve and set the session cookies. | ||||
| ``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:: | ||||
|  | ||||
| @@ -156,7 +154,7 @@ Or you can set the ``session_cookies`` on your initial login. | ||||
|     client = Client('<email>', '<password>', session_cookies=session_cookies) | ||||
|  | ||||
| .. warning:: | ||||
|     You session cookies can be just as valueable as you password, so store them with equal care | ||||
|     You session cookies can be just as valuable as you password, so store them with equal care | ||||
|  | ||||
|  | ||||
| .. _intro_events: | ||||
| @@ -164,13 +162,13 @@ Or you can set the ``session_cookies`` on your initial login. | ||||
| Listening & Events | ||||
| ------------------ | ||||
|  | ||||
| To use the listening functions `fbchat` offers (like :func:`Client.listen`), | ||||
| 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` | ||||
|     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:: | ||||
|  | ||||
| @@ -194,7 +192,7 @@ The change was in the parameters that our `onMessage` method took: ``message_obj | ||||
| and ``mid``, ``ts``, ``metadata`` and ``msg`` got removed, but the function still works, since we included ``**kwargs`` | ||||
|  | ||||
| .. note:: | ||||
|     Therefore, for both backwards and forwards compatability, | ||||
|     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 | ||||
|   | ||||
| @@ -5,21 +5,20 @@ pushd %~dp0 | ||||
| REM Command file for Sphinx documentation | ||||
|  | ||||
| if "%SPHINXBUILD%" == "" ( | ||||
| 	set SPHINXBUILD=python -msphinx | ||||
| 	set SPHINXBUILD=sphinx-build | ||||
| ) | ||||
| set SOURCEDIR=. | ||||
| set BUILDDIR=_build | ||||
| set SPHINXPROJ=fbchat | ||||
|  | ||||
| if "%1" == "" goto help | ||||
|  | ||||
| %SPHINXBUILD% >NUL 2>NUL | ||||
| if errorlevel 9009 ( | ||||
| 	echo. | ||||
| 	echo.The Sphinx module was not found. Make sure you have Sphinx installed, | ||||
| 	echo.then set the SPHINXBUILD environment variable to point to the full | ||||
| 	echo.path of the 'sphinx-build' executable. Alternatively you may add the | ||||
| 	echo.Sphinx directory to PATH. | ||||
| 	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/ | ||||
|   | ||||
							
								
								
									
										3
									
								
								docs/spelling/fixes.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								docs/spelling/fixes.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| premade | ||||
| todo | ||||
| emoji | ||||
							
								
								
									
										3
									
								
								docs/spelling/names.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								docs/spelling/names.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| Facebook | ||||
| GraphQL | ||||
| GitHub | ||||
							
								
								
									
										14
									
								
								docs/spelling/technical.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										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 | ||||
| @@ -1,5 +1,3 @@ | ||||
| .. highlight:: sh | ||||
| .. module:: fbchat | ||||
| .. _testing: | ||||
|  | ||||
| Testing | ||||
| @@ -15,7 +13,9 @@ To use the tests, copy ``tests/data.json`` to ``tests/my_data.json`` or type the | ||||
| 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:: | ||||
| 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 | ||||
|  | ||||
| @@ -23,7 +23,3 @@ If you only want to execute specific tests, pass the function names in the comma | ||||
|  | ||||
|     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) | ||||
|  | ||||
| .. automodule:: tests | ||||
|     :members: TestFbchat | ||||
|     :undoc-members: TestFbchat | ||||
|   | ||||
| @@ -1,5 +1,3 @@ | ||||
| .. highlight:: python | ||||
| .. module:: fbchat | ||||
| .. _todo: | ||||
|  | ||||
| Todo | ||||
| @@ -11,11 +9,11 @@ This page will be periodically updated to show missing features and documentatio | ||||
| Missing Functionality | ||||
| --------------------- | ||||
|  | ||||
| - Implement Client.searchForMessage | ||||
|     - This will use the graphql request API | ||||
| - Implement ``Client.searchForMessage`` | ||||
|     - This will use the GraphQL request API | ||||
| - Implement chatting with pages properly | ||||
| - Write better FAQ | ||||
| - Explain usage of graphql | ||||
| - Explain usage of GraphQL | ||||
|  | ||||
|  | ||||
| Documentation | ||||
|   | ||||
| @@ -3,10 +3,10 @@ | ||||
| from fbchat import Client | ||||
| from fbchat.models import * | ||||
|  | ||||
| client = Client('<email>', '<password>') | ||||
| client = Client("<email>", "<password>") | ||||
|  | ||||
| print('Own id: {}'.format(client.uid)) | ||||
| print("Own id: {}".format(client.uid)) | ||||
|  | ||||
| client.send(Message(text='Hi me!'), thread_id=client.uid, thread_type=ThreadType.USER) | ||||
| client.send(Message(text="Hi me!"), thread_id=client.uid, thread_type=ThreadType.USER) | ||||
|  | ||||
| client.logout() | ||||
|   | ||||
| @@ -14,5 +14,6 @@ class EchoBot(Client): | ||||
|         if author_id != self.uid: | ||||
|             self.send(message_object, thread_id=thread_id, thread_type=thread_type) | ||||
|  | ||||
|  | ||||
| client = EchoBot("<email>", "<password>") | ||||
| client.listen() | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| # -*- coding: UTF-8 -*- | ||||
|  | ||||
| from itertools import islice | ||||
| from fbchat import Client | ||||
| from fbchat.models import * | ||||
|  | ||||
| client = Client('<email>', '<password>') | ||||
| client = Client("<email>", "<password>") | ||||
|  | ||||
| # Fetches a list of all users you're currently chatting with, as `User` objects | ||||
| users = client.fetchAllUsers() | ||||
| @@ -13,9 +14,9 @@ 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>'] | ||||
| 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>') | ||||
| 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])) | ||||
| @@ -23,9 +24,9 @@ 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] | ||||
| user = client.searchForUsers("<name of user>")[0] | ||||
|  | ||||
| print('user ID: {}'.format(user.uid)) | ||||
| 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)) | ||||
| @@ -40,7 +41,7 @@ print("Threads: {}".format(threads)) | ||||
|  | ||||
|  | ||||
| # Gets the last 10 messages sent to the thread | ||||
| messages = client.fetchThreadMessages(thread_id='<thread id>', limit=10) | ||||
| messages = client.fetchThreadMessages(thread_id="<thread id>", limit=10) | ||||
| # Since the message come in reversed order, reverse them | ||||
| messages.reverse() | ||||
|  | ||||
| @@ -50,15 +51,21 @@ for message in messages: | ||||
|  | ||||
|  | ||||
| # If we have a thread id, we can use `fetchThreadInfo` to fetch a `Thread` object | ||||
| thread = client.fetchThreadInfo('<thread id>')['<thread id>'] | ||||
| 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] | ||||
| 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) | ||||
|   | ||||
| @@ -5,57 +5,89 @@ from fbchat.models import * | ||||
|  | ||||
| client = Client("<email>", "<password>") | ||||
|  | ||||
| thread_id = '1234567890' | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| # 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) | ||||
|     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) | ||||
|     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) | ||||
|     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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| client.changeThreadEmoji("👍", thread_id=thread_id) | ||||
|  | ||||
| # Will react to a message with a 😍 emoji | ||||
| client.reactToMessage('<message id>', MessageReaction.LOVE) | ||||
| client.reactToMessage("<message id>", MessageReaction.LOVE) | ||||
|   | ||||
| @@ -4,28 +4,33 @@ from fbchat import log, Client | ||||
| from fbchat.models import * | ||||
|  | ||||
| # Change this to your group id | ||||
| old_thread_id = '1234567890' | ||||
| old_thread_id = "1234567890" | ||||
|  | ||||
| # Change these to match your liking | ||||
| old_color = ThreadColor.MESSENGER_BLUE | ||||
| old_emoji = '👍' | ||||
| old_title = 'Old group chat name' | ||||
| 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" | ||||
|     "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)) | ||||
|             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)) | ||||
|             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): | ||||
| @@ -36,19 +41,43 @@ class KeepBot(Client): | ||||
|  | ||||
|     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: | ||||
|         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) | ||||
|             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, | ||||
|             ) | ||||
|  | ||||
|     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() | ||||
|   | ||||
| @@ -3,15 +3,23 @@ | ||||
| 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)) | ||||
|         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) | ||||
|             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() | ||||
|   | ||||
| @@ -4,21 +4,22 @@ | ||||
| :copyright: (c) 2015 - 2019 by Taehoon Kim | ||||
| :license: BSD 3-Clause, see LICENSE for more details. | ||||
| """ | ||||
|  | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from .client import * | ||||
| # These imports are far too general, but they're needed for backwards compatbility. | ||||
| from .models import * | ||||
|  | ||||
| __title__ = 'fbchat' | ||||
| __version__ = '1.6.1' | ||||
| __description__ = 'Facebook Chat (Messenger) for Python' | ||||
| from ._client import Client | ||||
| from ._util import log  # TODO: Remove this (from examples too) | ||||
|  | ||||
| __copyright__ = 'Copyright 2015 - 2019 by Taehoon Kim' | ||||
| __license__ = 'BSD 3-Clause' | ||||
| __title__ = "fbchat" | ||||
| __version__ = "1.9.6" | ||||
| __description__ = "Facebook Chat (Messenger) for Python" | ||||
|  | ||||
| __author__ = 'Taehoon Kim; Moreels Pieter-Jan; Mads Marquart' | ||||
| __email__ = 'carpedm20@gmail.com' | ||||
| __copyright__ = "Copyright 2015 - 2019 by Taehoon Kim" | ||||
| __license__ = "BSD 3-Clause" | ||||
|  | ||||
| __all__ = [ | ||||
|     'Client', | ||||
| ] | ||||
| __author__ = "Taehoon Kim; Moreels Pieter-Jan; Mads Marquart" | ||||
| __email__ = "carpedm20@gmail.com" | ||||
|  | ||||
| __all__ = ["Client"] | ||||
|   | ||||
							
								
								
									
										86
									
								
								fbchat/_attachment.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										3875
									
								
								fbchat/_client.py
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										26
									
								
								fbchat/_core.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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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
									
								
							
							
						
						
									
										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 | ||||
							
								
								
									
										3393
									
								
								fbchat/client.py
									
									
									
									
									
								
							
							
						
						
									
										3393
									
								
								fbchat/client.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,634 +0,0 @@ | ||||
| # -*- coding: UTF-8 -*- | ||||
|  | ||||
| from __future__ import unicode_literals | ||||
| import json | ||||
| import re | ||||
| from .models import * | ||||
| from .utils import * | ||||
|  | ||||
| # 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 graphql_color_to_enum(color): | ||||
|     if color is None: | ||||
|         return None | ||||
|     if not color: | ||||
|         return ThreadColor.MESSENGER_BLUE | ||||
|     color = color[2:]  # Strip the alpha value | ||||
|     color_value = '#{}'.format(color.lower()) | ||||
|     return enum_extend_if_invalid(ThreadColor, color_value) | ||||
|  | ||||
| def get_customization_info(thread): | ||||
|     if thread is None or thread.get('customization_info') is None: | ||||
|         return {} | ||||
|     info = thread['customization_info'] | ||||
|  | ||||
|     rtn = { | ||||
|         'emoji': info.get('emoji'), | ||||
|         'color': graphql_color_to_enum(info.get('outgoing_bubble_color')) | ||||
|     } | ||||
|     if thread.get('thread_type') == 'GROUP' or thread.get('is_group_thread') or thread.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 = thread.get('thread_key', {}).get('other_user_id') or thread.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 graphql_to_sticker(s): | ||||
|     if not s: | ||||
|         return None | ||||
|     sticker = Sticker( | ||||
|         uid=s['id'] | ||||
|     ) | ||||
|     if s.get('pack'): | ||||
|         sticker.pack = s['pack'].get('id') | ||||
|     if s.get('sprite_image'): | ||||
|         sticker.is_animated = True | ||||
|         sticker.medium_sprite_image = s['sprite_image'].get('uri') | ||||
|         sticker.large_sprite_image = s['sprite_image_2x'].get('uri') | ||||
|         sticker.frames_per_row = s.get('frames_per_row') | ||||
|         sticker.frames_per_col = s.get('frames_per_column') | ||||
|         sticker.frame_rate = s.get('frame_rate') | ||||
|     sticker.url = s.get('url') | ||||
|     sticker.width = s.get('width') | ||||
|     sticker.height = s.get('height') | ||||
|     if s.get('label'): | ||||
|         sticker.label = s['label'] | ||||
|     return sticker | ||||
|  | ||||
| def graphql_to_attachment(a): | ||||
|     _type = a['__typename'] | ||||
|     if _type in ['MessageImage', 'MessageAnimatedImage']: | ||||
|         return ImageAttachment( | ||||
|             original_extension=a.get('original_extension') or (a['filename'].split('-')[0] if a.get('filename') else None), | ||||
|             width=a.get('original_dimensions', {}).get('width'), | ||||
|             height=a.get('original_dimensions', {}).get('height'), | ||||
|             is_animated=_type=='MessageAnimatedImage', | ||||
|             thumbnail_url=a.get('thumbnail', {}).get('uri'), | ||||
|             preview=a.get('preview') or a.get('preview_image'), | ||||
|             large_preview=a.get('large_preview'), | ||||
|             animated_preview=a.get('animated_image'), | ||||
|             uid=a.get('legacy_attachment_id') | ||||
|         ) | ||||
|     elif _type == 'MessageVideo': | ||||
|         return VideoAttachment( | ||||
|             width=a.get('original_dimensions', {}).get('width'), | ||||
|             height=a.get('original_dimensions', {}).get('height'), | ||||
|             duration=a.get('playable_duration_in_ms'), | ||||
|             preview_url=a.get('playable_url'), | ||||
|             small_image=a.get('chat_image'), | ||||
|             medium_image=a.get('inbox_image'), | ||||
|             large_image=a.get('large_image'), | ||||
|             uid=a.get('legacy_attachment_id') | ||||
|         ) | ||||
|     elif _type == 'MessageAudio': | ||||
|         return AudioAttachment( | ||||
|             filename=a.get('filename'), | ||||
|             url=a.get('playable_url'), | ||||
|             duration=a.get('playable_duration_in_ms'), | ||||
|             audio_type=a.get('audio_type') | ||||
|         ) | ||||
|     elif _type == 'MessageFile': | ||||
|         return FileAttachment( | ||||
|             url=a.get('url'), | ||||
|             name=a.get('filename'), | ||||
|             is_malicious=a.get('is_malicious'), | ||||
|             uid=a.get('message_file_fbid') | ||||
|         ) | ||||
|     else: | ||||
|         return Attachment( | ||||
|             uid=a.get('legacy_attachment_id') | ||||
|         ) | ||||
|  | ||||
| def graphql_to_extensible_attachment(a): | ||||
|     story = a.get('story_attachment') | ||||
|     if story: | ||||
|         target = story.get('target') | ||||
|         if target: | ||||
|             _type = target['__typename'] | ||||
|             if _type == 'MessageLocation': | ||||
|                 latitude, longitude = get_url_parameter(get_url_parameter(story['url'], 'u'), 'where1').split(", ") | ||||
|                 rtn = LocationAttachment( | ||||
|                     uid=int(story['deduplication_key']), | ||||
|                     latitude=float(latitude), | ||||
|                     longitude=float(longitude), | ||||
|                 ) | ||||
|                 if story['media']: | ||||
|                     rtn.image_url = story['media']['image']['uri'] | ||||
|                     rtn.image_width = story['media']['image']['width'] | ||||
|                     rtn.image_height = story['media']['image']['height'] | ||||
|                 rtn.url = story['url'] | ||||
|                 return rtn | ||||
|             elif _type == 'MessageLiveLocation': | ||||
|                 rtn = LiveLocationAttachment( | ||||
|                     uid=int(story['target']['live_location_id']), | ||||
|                     latitude=story['target']['coordinate']['latitude'] if story['target'].get('coordinate') else None, | ||||
|                     longitude=story['target']['coordinate']['longitude'] if story['target'].get('coordinate') else None, | ||||
|                     name=story['title_with_entities']['text'], | ||||
|                     expiration_time=story['target']['expiration_time'] if story['target'].get('expiration_time') else None, | ||||
|                     is_expired=story['target']['is_expired'], | ||||
|                 ) | ||||
|                 if story['media']: | ||||
|                     rtn.image_url = story['media']['image']['uri'] | ||||
|                     rtn.image_width = story['media']['image']['width'] | ||||
|                     rtn.image_height = story['media']['image']['height'] | ||||
|                 rtn.url = story['url'] | ||||
|                 return rtn | ||||
|             elif _type in ['ExternalUrl', 'Story']: | ||||
|                 return ShareAttachment( | ||||
|                     uid=a.get('legacy_attachment_id'), | ||||
|                     author=story['target']['actors'][0]['id'] if story['target'].get('actors') else None, | ||||
|                     url=story['url'], | ||||
|                     original_url=get_url_parameter(story['url'], 'u') if "/l.php?u=" in story['url'] else story['url'], | ||||
|                     title=story['title_with_entities'].get('text'), | ||||
|                     description=story['description'].get('text') if story.get('description') else None, | ||||
|                     source=story['source']['text'], | ||||
|                     image_url=story['media']['image']['uri'] if story.get('media') else None, | ||||
|                     original_image_url=(get_url_parameter(story['media']['image']['uri'], 'url') if "/safe_image.php" in story['media']['image']['uri'] else story['media']['image']['uri']) if story.get('media') else None, | ||||
|                     image_width=story['media']['image']['width'] if story.get('media') else None, | ||||
|                     image_height=story['media']['image']['height'] if story.get('media') else None, | ||||
|                     attachments=[graphql_to_subattachment(attachment) for attachment in story.get('subattachments')], | ||||
|                 ) | ||||
|         else: | ||||
|             return UnsentMessage( | ||||
|                 uid=a.get('legacy_attachment_id'), | ||||
|             ) | ||||
|  | ||||
|  | ||||
| def graphql_to_subattachment(a): | ||||
|     _type = a['target']['__typename'] | ||||
|     if _type == 'Video': | ||||
|         return VideoAttachment( | ||||
|             duration=a['media'].get('playable_duration_in_ms'), | ||||
|             preview_url=a['media'].get('playable_url'), | ||||
|             medium_image=a['media'].get('image'), | ||||
|             uid=a['target'].get('video_id'), | ||||
|         ) | ||||
|  | ||||
| def graphql_to_live_location(a): | ||||
|     return LiveLocationAttachment( | ||||
|         uid=a['id'], | ||||
|         latitude=a['coordinate']['latitude'] / (10 ** 8) if not a.get('stopReason') else None, | ||||
|         longitude=a['coordinate']['longitude'] / (10 ** 8)  if not a.get('stopReason') else None, | ||||
|         name=a.get('locationTitle'), | ||||
|         expiration_time=a['expirationTime'], | ||||
|         is_expired=bool(a.get('stopReason')), | ||||
|     ) | ||||
|  | ||||
| def graphql_to_poll(a): | ||||
|     rtn = Poll( | ||||
|         title=a.get('title') if a.get('title') else a.get('text'), | ||||
|         options=[graphql_to_poll_option(m) for m in a.get('options')] | ||||
|     ) | ||||
|     rtn.uid = int(a["id"]) | ||||
|     rtn.options_count = a.get("total_count") | ||||
|     return rtn | ||||
|  | ||||
| def graphql_to_poll_option(a): | ||||
|     if a.get('viewer_has_voted') is None: | ||||
|         vote = None | ||||
|     elif isinstance(a['viewer_has_voted'], bool): | ||||
|         vote = a['viewer_has_voted'] | ||||
|     else: | ||||
|         vote = a['viewer_has_voted'] == 'true' | ||||
|     rtn = PollOption( | ||||
|         text=a.get('text'), | ||||
|         vote=vote | ||||
|     ) | ||||
|     rtn.uid = int(a["id"]) | ||||
|     rtn.voters = [m.get('node').get('id') for m in a.get('voters').get('edges')] if isinstance(a.get('voters'), dict) else a.get('voters') | ||||
|     rtn.votes_count = a.get('voters').get('count') if isinstance(a.get('voters'), dict) else a.get('total_count') | ||||
|     return rtn | ||||
|  | ||||
| def graphql_to_plan(a): | ||||
|     if a.get('event_members'): | ||||
|         rtn = Plan( | ||||
|             time=a.get('event_time'), | ||||
|             title=a.get('title'), | ||||
|             location=a.get('location_name') | ||||
|         ) | ||||
|         if a.get('location_id') != 0: | ||||
|             rtn.location_id = str(a.get('location_id')) | ||||
|         rtn.uid = a.get('oid') | ||||
|         rtn.author_id = a.get('creator_id') | ||||
|         guests = a.get("event_members") | ||||
|         rtn.going = [uid for uid in guests if guests[uid] == "GOING"] | ||||
|         rtn.declined = [uid for uid in guests if guests[uid] == "DECLINED"] | ||||
|         rtn.invited = [uid for uid in guests if guests[uid] == "INVITED"] | ||||
|         return rtn | ||||
|     elif a.get('id') is None: | ||||
|         rtn = Plan( | ||||
|             time=a.get('event_time'), | ||||
|             title=a.get('event_title'), | ||||
|             location=a.get('event_location_name'), | ||||
|             location_id=a.get('event_location_id') | ||||
|         ) | ||||
|         rtn.uid = a.get('event_id') | ||||
|         rtn.author_id = a.get('event_creator_id') | ||||
|         guests = json.loads(a.get('guest_state_list')) | ||||
|     else: | ||||
|         rtn = Plan( | ||||
|             time=a.get('time'), | ||||
|             title=a.get('event_title'), | ||||
|             location=a.get('location_name') | ||||
|         ) | ||||
|         rtn.uid = a.get('id') | ||||
|         rtn.author_id = a.get('lightweight_event_creator').get('id') | ||||
|         guests = a.get('event_reminder_members').get('edges') | ||||
|     rtn.going = [m.get('node').get('id') for m in guests if m.get('guest_list_state') == "GOING"] | ||||
|     rtn.declined = [m.get('node').get('id') for m in guests if m.get('guest_list_state') == "DECLINED"] | ||||
|     rtn.invited = [m.get('node').get('id') for m in guests if m.get('guest_list_state') == "INVITED"] | ||||
|     return rtn | ||||
|  | ||||
| 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 | ||||
|  | ||||
| def graphql_to_message(message): | ||||
|     if message.get('message_sender') is None: | ||||
|         message['message_sender'] = {} | ||||
|     if message.get('message') is None: | ||||
|         message['message'] = {} | ||||
|     rtn = Message( | ||||
|         text=message.get('message').get('text'), | ||||
|         mentions=[Mention(m.get('entity', {}).get('id'), offset=m.get('offset'), length=m.get('length')) for m in message.get('message').get('ranges', [])], | ||||
|         emoji_size=get_emojisize_from_tags(message.get('tags_list')), | ||||
|         sticker=graphql_to_sticker(message.get('sticker')) | ||||
|     ) | ||||
|     rtn.uid = str(message.get('message_id')) | ||||
|     rtn.author = str(message.get('message_sender').get('id')) | ||||
|     rtn.timestamp = message.get('timestamp_precise') | ||||
|     rtn.unsent = False | ||||
|     if message.get('unread') is not None: | ||||
|         rtn.is_read = not message['unread'] | ||||
|     rtn.reactions = { | ||||
|         str(r['user']['id']): enum_extend_if_invalid(MessageReaction, r['reaction']) | ||||
|         for r in message.get('message_reactions') | ||||
|     } | ||||
|     if message.get('blob_attachments') is not None: | ||||
|         rtn.attachments = [graphql_to_attachment(attachment) for attachment in message['blob_attachments']] | ||||
|     if message.get('platform_xmd_encoded'): | ||||
|         quick_replies = json.loads(message['platform_xmd_encoded']).get('quick_replies') | ||||
|         if isinstance(quick_replies, list): | ||||
|             rtn.quick_replies = [graphql_to_quick_reply(q) for q in quick_replies] | ||||
|         elif isinstance(quick_replies, dict): | ||||
|             rtn.quick_replies = [graphql_to_quick_reply(quick_replies, is_response=True)] | ||||
|     if message.get('extensible_attachment') is not None: | ||||
|         attachment = graphql_to_extensible_attachment(message['extensible_attachment']) | ||||
|         if isinstance(attachment, UnsentMessage): | ||||
|             rtn.unsent = True | ||||
|         elif attachment: | ||||
|             rtn.attachments.append(attachment) | ||||
|     return rtn | ||||
|  | ||||
| def graphql_to_user(user): | ||||
|     if user.get('profile_picture') is None: | ||||
|         user['profile_picture'] = {} | ||||
|     c_info = get_customization_info(user) | ||||
|     plan = None | ||||
|     if user.get('event_reminders'): | ||||
|         plan = graphql_to_plan(user['event_reminders']['nodes'][0]) if user['event_reminders'].get('nodes') else None | ||||
|     return User( | ||||
|         user['id'], | ||||
|         url=user.get('url'), | ||||
|         first_name=user.get('first_name'), | ||||
|         last_name=user.get('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['profile_picture'].get('uri'), | ||||
|         name=user.get('name'), | ||||
|         message_count=user.get('messages_count'), | ||||
|         plan=plan, | ||||
|     ) | ||||
|  | ||||
| def graphql_to_thread(thread): | ||||
|     if thread['thread_type'] == 'GROUP': | ||||
|         return graphql_to_group(thread) | ||||
|     elif thread['thread_type'] == 'ONE_TO_ONE': | ||||
|         if thread.get('big_image_src') is None: | ||||
|             thread['big_image_src'] = {} | ||||
|         c_info = get_customization_info(thread) | ||||
|         participants = [node['messaging_actor'] for node in thread['all_participants']['nodes']] | ||||
|         user = next(p for p in participants if p['id'] == thread['thread_key']['other_user_id']) | ||||
|         last_message_timestamp = None | ||||
|         if 'last_message' in thread: | ||||
|             last_message_timestamp = thread['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 thread.get('event_reminders'): | ||||
|             plan = graphql_to_plan(thread['event_reminders']['nodes'][0]) if thread['event_reminders'].get('nodes') else None | ||||
|  | ||||
|         return User( | ||||
|             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=thread.get('messages_count'), | ||||
|             last_message_timestamp=last_message_timestamp, | ||||
|             plan=plan, | ||||
|         ) | ||||
|     else: | ||||
|         raise FBchatException('Unknown thread type: {}, with data: {}'.format(thread.get('thread_type'), thread)) | ||||
|  | ||||
| def graphql_to_group(group): | ||||
|     if group.get('image') is None: | ||||
|         group['image'] = {} | ||||
|     c_info = get_customization_info(group) | ||||
|     last_message_timestamp = None | ||||
|     if 'last_message' in group: | ||||
|         last_message_timestamp = group['last_message']['nodes'][0]['timestamp_precise'] | ||||
|     plan = None | ||||
|     if group.get('event_reminders'): | ||||
|         plan = graphql_to_plan(group['event_reminders']['nodes'][0]) if group['event_reminders'].get('nodes') else None | ||||
|     return Group( | ||||
|         group['thread_key']['thread_fbid'], | ||||
|         participants=set([node['messaging_actor']['id'] for node in group['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 group.get('thread_admins')]), | ||||
|         approval_mode = bool(group.get('approval_mode')) if group.get('approval_mode') is not None else None, | ||||
|         approval_requests = set(node["requester"]['id'] for node in group['group_approval_queue']['nodes']) if group.get('group_approval_queue') else None, | ||||
|         join_link = group['joinable_mode'].get('link'), | ||||
|         photo=group['image'].get('uri'), | ||||
|         name=group.get('name'), | ||||
|         message_count=group.get('messages_count'), | ||||
|         last_message_timestamp=last_message_timestamp, | ||||
|         plan=plan, | ||||
|     ) | ||||
|  | ||||
| def graphql_to_page(page): | ||||
|     if page.get('profile_picture') is None: | ||||
|         page['profile_picture'] = {} | ||||
|     if page.get('city') is None: | ||||
|         page['city'] = {} | ||||
|     plan = None | ||||
|     if page.get('event_reminders'): | ||||
|         plan = graphql_to_plan(page['event_reminders']['nodes'][0]) if page['event_reminders'].get('nodes') else None | ||||
|     return Page( | ||||
|         page['id'], | ||||
|         url=page.get('url'), | ||||
|         city=page.get('city').get('name'), | ||||
|         category=page.get('category_type'), | ||||
|         photo=page['profile_picture'].get('uri'), | ||||
|         name=page.get('name'), | ||||
|         message_count=page.get('messages_count'), | ||||
|         plan=plan, | ||||
|     ) | ||||
|  | ||||
| def graphql_queries_to_json(*queries): | ||||
|     """ | ||||
|     Queries should be a list of GraphQL objects | ||||
|     """ | ||||
|     rtn = {} | ||||
|     for i, query in enumerate(queries): | ||||
|         rtn['q{}'.format(i)] = query.value | ||||
|     return json.dumps(rtn) | ||||
|  | ||||
| def graphql_response_to_json(content): | ||||
|     content = strip_to_json(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 | ||||
|         check_json(x) | ||||
|         [(key, value)] = x.items() | ||||
|         check_json(value) | ||||
|         if 'response' in value: | ||||
|             rtn[int(key[1:])] = value['response'] | ||||
|         else: | ||||
|             rtn[int(key[1:])] = value['data'] | ||||
|  | ||||
|     log.debug(rtn) | ||||
|  | ||||
|     return rtn | ||||
|  | ||||
| class GraphQL(object): | ||||
|     def __init__(self, query=None, doc_id=None, params=None): | ||||
|         if params is None: | ||||
|             params = {} | ||||
|         if query is not None: | ||||
|             self.value = { | ||||
|                 'priority': 0, | ||||
|                 'q': query, | ||||
|                 'query_params': params | ||||
|             } | ||||
|         elif doc_id is not None: | ||||
|             self.value = { | ||||
|                 'doc_id': doc_id, | ||||
|                 'query_params': params | ||||
|             } | ||||
|         else: | ||||
|             raise FBchatUserError('A query or doc_id must be specified') | ||||
|  | ||||
|  | ||||
|     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 | ||||
							
								
								
									
										821
									
								
								fbchat/models.py
									
									
									
									
									
								
							
							
						
						
									
										821
									
								
								fbchat/models.py
									
									
									
									
									
								
							| @@ -1,800 +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 | ||||
| import aenum | ||||
| from string import Formatter | ||||
|  | ||||
|  | ||||
| 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 (eg. 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 FBchatUserError(FBchatException): | ||||
|     """Thrown by fbchat when wrong values are entered""" | ||||
|  | ||||
| class Thread(object): | ||||
|     #: The unique identifier of the thread. Can be used a `thread_id`. See :ref:`intro_threads` for more info | ||||
|     uid = None | ||||
|     #: Specifies the type of thread. Can be used a `thread_type`. See :ref:`intro_threads` for more info | ||||
|     type = None | ||||
|     #: A url to the thread's picture | ||||
|     photo = None | ||||
|     #: The name of the thread | ||||
|     name = None | ||||
|     #: Timestamp of last message | ||||
|     last_message_timestamp = None | ||||
|     #: Number of messages in the thread | ||||
|     message_count = None | ||||
|     #: Set :class:`Plan` | ||||
|     plan = None | ||||
|     def __init__(self, _type, uid, photo=None, name=None, last_message_timestamp=None, message_count=None, plan=None): | ||||
|         """Represents a Facebook thread""" | ||||
|         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 | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return self.__unicode__() | ||||
|  | ||||
|     def __unicode__(self): | ||||
|         return '<{} {} ({})>'.format(self.type.name, self.name, self.uid) | ||||
|  | ||||
|  | ||||
| class User(Thread): | ||||
|     #: The profile url | ||||
|     url = None | ||||
|     #: The users first name | ||||
|     first_name = None | ||||
|     #: The users last name | ||||
|     last_name = None | ||||
|     #: Whether the user and the client are friends | ||||
|     is_friend = None | ||||
|     #: The user's gender | ||||
|     gender = None | ||||
|     #: From 0 to 1. How close the client is to the user | ||||
|     affinity = None | ||||
|     #: The user's nickname | ||||
|     nickname = None | ||||
|     #: The clients nickname, as seen by the user | ||||
|     own_nickname = None | ||||
|     #: A :class:`ThreadColor`. The message color | ||||
|     color = None | ||||
|     #: The default emoji | ||||
|     emoji = 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): | ||||
|         """Represents a Facebook user. Inherits `Thread`""" | ||||
|         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 | ||||
|  | ||||
|  | ||||
| class Group(Thread): | ||||
|     #: Unique list (set) of the group thread's participant user IDs | ||||
|     participants = None | ||||
|     #: A dict, containing user nicknames mapped to their IDs | ||||
|     nicknames = None | ||||
|     #: A :class:`ThreadColor`. The groups's message color | ||||
|     color = None | ||||
|     #: The groups's default emoji | ||||
|     emoji = None | ||||
|     # Set containing user IDs of thread admins | ||||
|     admins = None | ||||
|     # True if users need approval to join | ||||
|     approval_mode = None | ||||
|     # Set containing user IDs requesting to join | ||||
|     approval_requests = None | ||||
|     # Link for joining group | ||||
|     join_link = 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): | ||||
|         """Represents a Facebook group. Inherits `Thread`""" | ||||
|         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 | ||||
|  | ||||
|  | ||||
| class Room(Group): | ||||
|     # True is room is not discoverable | ||||
|     privacy_mode = None | ||||
|  | ||||
|     def __init__(self, uid, privacy_mode=None, **kwargs): | ||||
|         """Deprecated. Use :class:`Group` instead""" | ||||
|         super(Room, self).__init__(uid, **kwargs) | ||||
|         self.type = ThreadType.ROOM | ||||
|         self.privacy_mode = privacy_mode | ||||
|  | ||||
|  | ||||
| class Page(Thread): | ||||
|     #: The page's custom url | ||||
|     url = None | ||||
|     #: The name of the page's location city | ||||
|     city = None | ||||
|     #: Amount of likes the page has | ||||
|     likes = None | ||||
|     #: Some extra information about the page | ||||
|     sub_title = None | ||||
|     #: The page's category | ||||
|     category = None | ||||
|  | ||||
|     def __init__(self, uid, url=None, city=None, likes=None, sub_title=None, category=None, **kwargs): | ||||
|         """Represents a Facebook page. Inherits `Thread`""" | ||||
|         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 | ||||
|  | ||||
|  | ||||
| class Message(object): | ||||
|     #: The actual message | ||||
|     text = None | ||||
|     #: A list of :class:`Mention` objects | ||||
|     mentions = None | ||||
|     #: A :class:`EmojiSize`. Size of a sent emoji | ||||
|     emoji_size = None | ||||
|     #: The message ID | ||||
|     uid = None | ||||
|     #: ID of the sender | ||||
|     author = None | ||||
|     #: Timestamp of when the message was sent | ||||
|     timestamp = None | ||||
|     #: Whether the message is read | ||||
|     is_read = None | ||||
|     #: A list of pepole IDs who read the message, works only with :func:`fbchat.Client.fetchThreadMessages` | ||||
|     read_by = None | ||||
|     #: A dict with user's IDs as keys, and their :class:`MessageReaction` as values | ||||
|     reactions = None | ||||
|     #: The actual message | ||||
|     text = None | ||||
|     #: A :class:`Sticker` | ||||
|     sticker = None | ||||
|     #: A list of attachments | ||||
|     attachments = None | ||||
|     #: A list of :class:`QuickReply` | ||||
|     quick_replies = None | ||||
|     #: Whether the message is unsent (deleted for everyone) | ||||
|     unsent = None | ||||
|  | ||||
|     def __init__(self, text=None, mentions=None, emoji_size=None, sticker=None, attachments=None, quick_replies=None): | ||||
|         """Represents a Facebook message""" | ||||
|         self.text = text | ||||
|         if mentions is None: | ||||
|             mentions = [] | ||||
|         self.mentions = mentions | ||||
|         self.emoji_size = emoji_size | ||||
|         self.sticker = sticker | ||||
|         if attachments is None: | ||||
|             attachments = [] | ||||
|         self.attachments = attachments | ||||
|         if quick_replies is None: | ||||
|             quick_replies = [] | ||||
|         self.quick_replies = quick_replies | ||||
|         self.reactions = {} | ||||
|         self.read_by = [] | ||||
|         self.deleted = False | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return self.__unicode__() | ||||
|  | ||||
|     def __unicode__(self): | ||||
|         return '<Message ({}): {}, mentions={} emoji_size={} attachments={}>'.format(self.uid, repr(self.text), self.mentions, self.emoji_size, self.attachments) | ||||
|  | ||||
|     @classmethod | ||||
|     def formatMentions(cls, text, *args, **kwargs): | ||||
|         """Like `str.format`, but takes tuples with a thread id and text instead. | ||||
|  | ||||
|         Returns 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 | ||||
|  | ||||
| class Attachment(object): | ||||
|     #: The attachment ID | ||||
|     uid = None | ||||
|  | ||||
|     def __init__(self, uid=None): | ||||
|         """Represents a Facebook attachment""" | ||||
|         self.uid = uid | ||||
|  | ||||
| class UnsentMessage(Attachment): | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         """Represents an unsent message attachment""" | ||||
|         super(UnsentMessage, self).__init__(*args, **kwargs) | ||||
|  | ||||
| class Sticker(Attachment): | ||||
|     #: The sticker-pack's ID | ||||
|     pack = None | ||||
|     #: Whether the sticker is animated | ||||
|     is_animated = False | ||||
|  | ||||
|     # If the sticker is animated, the following should be present | ||||
|     #: URL to a medium spritemap | ||||
|     medium_sprite_image = None | ||||
|     #: URL to a large spritemap | ||||
|     large_sprite_image = None | ||||
|     #: The amount of frames present in the spritemap pr. row | ||||
|     frames_per_row = None | ||||
|     #: The amount of frames present in the spritemap pr. coloumn | ||||
|     frames_per_col = None | ||||
|     #: The frame rate the spritemap is intended to be played in | ||||
|     frame_rate = None | ||||
|  | ||||
|     #: URL to the sticker's image | ||||
|     url = None | ||||
|     #: Width of the sticker | ||||
|     width = None | ||||
|     #: Height of the sticker | ||||
|     height = None | ||||
|     #: The sticker's label/name | ||||
|     label = None | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         """Represents a Facebook sticker that has been sent to a Facebook thread as an attachment""" | ||||
|         super(Sticker, self).__init__(*args, **kwargs) | ||||
|  | ||||
| class ShareAttachment(Attachment): | ||||
|     #: ID of the author of the shared post | ||||
|     author = None | ||||
|     #: Target URL | ||||
|     url = None | ||||
|     #: Original URL if Facebook redirects the URL | ||||
|     original_url = None | ||||
|     #: Title of the attachment | ||||
|     title = None | ||||
|     #: Description of the attachment | ||||
|     description = None | ||||
|     #: Name of the source | ||||
|     source = None | ||||
|     #: URL of the attachment image | ||||
|     image_url = None | ||||
|     #: URL of the original image if Facebook uses `safe_image` | ||||
|     original_image_url = None | ||||
|     #: Width of the image | ||||
|     image_width = None | ||||
|     #: Height of the image | ||||
|     image_height = None | ||||
|     #: List of additional attachments | ||||
|     attachments = None | ||||
|  | ||||
|     def __init__(self, author=None, url=None, original_url=None, title=None, description=None, source=None, image_url=None, original_image_url=None, image_width=None, image_height=None, attachments=None, **kwargs): | ||||
|         """Represents a shared item (eg. URL) that has been sent as a Facebook attachment""" | ||||
|         super(ShareAttachment, self).__init__(**kwargs) | ||||
|         self.author = author | ||||
|         self.url = url | ||||
|         self.original_url = original_url | ||||
|         self.title = title | ||||
|         self.description = description | ||||
|         self.source = source | ||||
|         self.image_url = image_url | ||||
|         self.original_image_url = original_image_url | ||||
|         self.image_width = image_width | ||||
|         self.image_height = image_height | ||||
|         if attachments is None: | ||||
|             attachments = [] | ||||
|         self.attachments = attachments | ||||
|  | ||||
| class LocationAttachment(Attachment): | ||||
|     #: Latidute of the location | ||||
|     latitude = None | ||||
|     #: Longitude of the location | ||||
|     longitude = None | ||||
|     #: URL of image showing the map of the location | ||||
|     image_url = None | ||||
|     #: Width of the image | ||||
|     image_width = None | ||||
|     #: Height of the image | ||||
|     image_height = None | ||||
|     #: URL to Bing maps with the location | ||||
|     url = None | ||||
|  | ||||
|     def __init__(self, latitude=None, longitude=None, **kwargs): | ||||
|         """Represents a user location""" | ||||
|         super(LocationAttachment, self).__init__(**kwargs) | ||||
|         self.latitude = latitude | ||||
|         self.longitude = longitude | ||||
|  | ||||
| class LiveLocationAttachment(LocationAttachment): | ||||
|     #: Name of the location | ||||
|     name = None | ||||
|     #: Timestamp when live location expires | ||||
|     expiration_time = None | ||||
|     #: True if live location is expired | ||||
|     is_expired = None | ||||
|  | ||||
|     def __init__(self, name=None, expiration_time=None, is_expired=None, **kwargs): | ||||
|         """Represents a live user location""" | ||||
|         super(LiveLocationAttachment, self).__init__(**kwargs) | ||||
|         self.expiration_time = expiration_time | ||||
|         self.is_expired = is_expired | ||||
|  | ||||
| class FileAttachment(Attachment): | ||||
|     #: Url where you can download the file | ||||
|     url = None | ||||
|     #: Size of the file in bytes | ||||
|     size = None | ||||
|     #: Name of the file | ||||
|     name = None | ||||
|     #: Whether Facebook determines that this file may be harmful | ||||
|     is_malicious = None | ||||
|  | ||||
|     def __init__(self, url=None, size=None, name=None, is_malicious=None, **kwargs): | ||||
|         """Represents a file that has been sent as a Facebook attachment""" | ||||
|         super(FileAttachment, self).__init__(**kwargs) | ||||
|         self.url = url | ||||
|         self.size = size | ||||
|         self.name = name | ||||
|         self.is_malicious = is_malicious | ||||
|  | ||||
| class AudioAttachment(Attachment): | ||||
|     #: Name of the file | ||||
|     filename = None | ||||
|     #: Url of the audio file | ||||
|     url = None | ||||
|     #: Duration of the audioclip in milliseconds | ||||
|     duration = None | ||||
|     #: Audio type | ||||
|     audio_type = None | ||||
|  | ||||
|     def __init__(self, filename=None, url=None, duration=None, audio_type=None, **kwargs): | ||||
|         """Represents an audio file that has been sent as a Facebook attachment""" | ||||
|         super(AudioAttachment, self).__init__(**kwargs) | ||||
|         self.filename = filename | ||||
|         self.url = url | ||||
|         self.duration = duration | ||||
|         self.audio_type = audio_type | ||||
|  | ||||
| class ImageAttachment(Attachment): | ||||
|     #: The extension of the original image (eg. 'png') | ||||
|     original_extension = None | ||||
|     #: Width of original image | ||||
|     width = None | ||||
|     #: Height of original image | ||||
|     height = None | ||||
|  | ||||
|     #: Whether the image is animated | ||||
|     is_animated = None | ||||
|  | ||||
|     #: URL to a thumbnail of the image | ||||
|     thumbnail_url = None | ||||
|  | ||||
|     #: URL to a medium preview of the image | ||||
|     preview_url = None | ||||
|     #: Width of the medium preview image | ||||
|     preview_width = None | ||||
|     #: Height of the medium preview image | ||||
|     preview_height = None | ||||
|  | ||||
|     #: URL to a large preview of the image | ||||
|     large_preview_url = None | ||||
|     #: Width of the large preview image | ||||
|     large_preview_width = None | ||||
|     #: Height of the large preview image | ||||
|     large_preview_height = None | ||||
|  | ||||
|     #: URL to an animated preview of the image (eg. for gifs) | ||||
|     animated_preview_url = None | ||||
|     #: Width of the animated preview image | ||||
|     animated_preview_width = None | ||||
|     #: Height of the animated preview image | ||||
|     animated_preview_height = 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): | ||||
|         """ | ||||
|         Represents an image that has been sent as a Facebook attachment | ||||
|         To retrieve the full image url, use: :func:`fbchat.Client.fetchImageUrl`, | ||||
|         and pass it the uid of the image attachment | ||||
|         """ | ||||
|         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') | ||||
|  | ||||
| class VideoAttachment(Attachment): | ||||
|     #: Size of the original video in bytes | ||||
|     size = None | ||||
|     #: Width of original video | ||||
|     width = None | ||||
|     #: Height of original video | ||||
|     height = None | ||||
|     #: Length of video in milliseconds | ||||
|     duration = None | ||||
|     #: URL to very compressed preview video | ||||
|     preview_url = None | ||||
|  | ||||
|     #: URL to a small preview image of the video | ||||
|     small_image_url = None | ||||
|     #: Width of the small preview image | ||||
|     small_image_width = None | ||||
|     #: Height of the small preview image | ||||
|     small_image_height = None | ||||
|  | ||||
|     #: URL to a medium preview image of the video | ||||
|     medium_image_url = None | ||||
|     #: Width of the medium preview image | ||||
|     medium_image_width = None | ||||
|     #: Height of the medium preview image | ||||
|     medium_image_height = None | ||||
|  | ||||
|     #: URL to a large preview image of the video | ||||
|     large_image_url = None | ||||
|     #: Width of the large preview image | ||||
|     large_image_width = None | ||||
|     #: Height of the large preview image | ||||
|     large_image_height = 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): | ||||
|         """Represents a video that has been sent as a Facebook attachment""" | ||||
|         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') | ||||
|  | ||||
|  | ||||
| class Mention(object): | ||||
|     #: The thread ID the mention is pointing at | ||||
|     thread_id = None | ||||
|     #: The character where the mention starts | ||||
|     offset = None | ||||
|     #: The length of the mention | ||||
|     length = None | ||||
|  | ||||
|     def __init__(self, thread_id, offset=0, length=10): | ||||
|         """Represents a @mention""" | ||||
|         self.thread_id = thread_id | ||||
|         self.offset = offset | ||||
|         self.length = length | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return self.__unicode__() | ||||
|  | ||||
|     def __unicode__(self): | ||||
|         return '<Mention {}: offset={} length={}>'.format(self.thread_id, self.offset, self.length) | ||||
|  | ||||
| class QuickReply(object): | ||||
|     #: Payload of the quick reply | ||||
|     payload = None | ||||
|     #: External payload for responses | ||||
|     external_payload = None | ||||
|     #: Additional data | ||||
|     data = None | ||||
|     #: Whether it's a response for a quick reply | ||||
|     is_response = None | ||||
|  | ||||
|     def __init__(self, payload=None, data=None, is_response=False): | ||||
|         """Represents a quick reply""" | ||||
|         self.payload = payload | ||||
|         self.data = data | ||||
|         self.is_response = is_response | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return self.__unicode__() | ||||
|  | ||||
|     def __unicode__(self): | ||||
|         return '<{}: payload={!r}>'.format(self.__class__.__name__, self.payload) | ||||
|  | ||||
| class QuickReplyText(QuickReply): | ||||
|     #: Title of the quick reply | ||||
|     title = None | ||||
|     #: URL of the quick reply image (optional) | ||||
|     image_url = None | ||||
|     #: Type of the quick reply | ||||
|     _type = "text" | ||||
|  | ||||
|     def __init__(self, title=None, image_url=None, **kwargs): | ||||
|         """Represents a text quick reply""" | ||||
|         super(QuickReplyText, self).__init__(**kwargs) | ||||
|         self.title = title | ||||
|         self.image_url = image_url | ||||
|  | ||||
| class QuickReplyLocation(QuickReply): | ||||
|     #: Type of the quick reply | ||||
|     _type = "location" | ||||
|  | ||||
|     def __init__(self, **kwargs): | ||||
|         """Represents a location quick reply (Doesn't work on mobile)""" | ||||
|         super(QuickReplyLocation, self).__init__(**kwargs) | ||||
|         self.is_response = False | ||||
|  | ||||
| class QuickReplyPhoneNumber(QuickReply): | ||||
|     #: URL of the quick reply image (optional) | ||||
|     image_url = None | ||||
|     #: Type of the quick reply | ||||
|     _type = "user_phone_number" | ||||
|  | ||||
|     def __init__(self, image_url=None, **kwargs): | ||||
|         """Represents a phone number quick reply (Doesn't work on mobile)""" | ||||
|         super(QuickReplyPhoneNumber, self).__init__(**kwargs) | ||||
|         self.image_url = image_url | ||||
|  | ||||
| class QuickReplyEmail(QuickReply): | ||||
|     #: URL of the quick reply image (optional) | ||||
|     image_url = None | ||||
|     #: Type of the quick reply | ||||
|     _type = "user_email" | ||||
|  | ||||
|     def __init__(self, image_url=None, **kwargs): | ||||
|         """Represents an email quick reply (Doesn't work on mobile)""" | ||||
|         super(QuickReplyEmail, self).__init__(**kwargs) | ||||
|         self.image_url = image_url | ||||
|  | ||||
| class Poll(object): | ||||
|     #: ID of the poll | ||||
|     uid = None | ||||
|     #: Title of the poll | ||||
|     title = None | ||||
|     #: List of :class:`PollOption`, can be fetched with :func:`fbchat.Client.fetchPollOptions` | ||||
|     options = None | ||||
|     #: Options count | ||||
|     options_count = None | ||||
|  | ||||
|     def __init__(self, title, options): | ||||
|         """Represents a poll""" | ||||
|         self.title = title | ||||
|         self.options = options | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return self.__unicode__() | ||||
|  | ||||
|     def __unicode__(self): | ||||
|         return '<Poll ({}): {} options={}>'.format(self.uid, repr(self.title), self.options) | ||||
|  | ||||
| class PollOption(object): | ||||
|     #: ID of the poll option | ||||
|     uid = None | ||||
|     #: Text of the poll option | ||||
|     text = None | ||||
|     #: Whether vote when creating or client voted | ||||
|     vote = None | ||||
|     #: ID of the users who voted for this poll option | ||||
|     voters = None | ||||
|     #: Votes count | ||||
|     votes_count = None | ||||
|  | ||||
|     def __init__(self, text, vote=False): | ||||
|         """Represents a poll option""" | ||||
|         self.text = text | ||||
|         self.vote = vote | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return self.__unicode__() | ||||
|  | ||||
|     def __unicode__(self): | ||||
|         return '<PollOption ({}): {} voters={}>'.format(self.uid, repr(self.text), self.voters) | ||||
|  | ||||
| class Plan(object): | ||||
|     #: ID of the plan | ||||
|     uid = None | ||||
|     #: Plan time (unix time stamp), only precise down to the minute | ||||
|     time = None | ||||
|     #: Plan title | ||||
|     title = None | ||||
|     #: Plan location name | ||||
|     location = None | ||||
|     #: Plan location ID | ||||
|     location_id = None | ||||
|     #: ID of the plan creator | ||||
|     author_id = None | ||||
|     #: List of the people IDs who will take part in the plan | ||||
|     going = None | ||||
|     #: List of the people IDs who won't take part in the plan | ||||
|     declined = None | ||||
|     #: List of the people IDs who are invited to the plan | ||||
|     invited = None | ||||
|  | ||||
|     def __init__(self, time, title, location=None, location_id=None): | ||||
|         """Represents a plan""" | ||||
|         self.time = int(time) | ||||
|         self.title = title | ||||
|         self.location = location or '' | ||||
|         self.location_id = location_id or '' | ||||
|         self.author_id = None | ||||
|         self.going = [] | ||||
|         self.declined = [] | ||||
|         self.invited = [] | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return self.__unicode__() | ||||
|  | ||||
|     def __unicode__(self): | ||||
|         return '<Plan ({}): {} time={}, location={}, location_id={}>'.format(self.uid, repr(self.title), self.time, repr(self.location), repr(self.location_id)) | ||||
|  | ||||
| class ActiveStatus(object): | ||||
|     #: Whether the user is active now | ||||
|     active = None | ||||
|     #: Timestamp when the user was last active | ||||
|     last_active = None | ||||
|     #: Whether the user is playing Messenger game now | ||||
|     in_game = None | ||||
|  | ||||
|     def __init__(self, active=None, last_active=None, in_game=None): | ||||
|         self.active = active | ||||
|         self.last_active = last_active | ||||
|         self.in_game = in_game | ||||
|  | ||||
|     def __repr__(self): | ||||
|         return self.__unicode__() | ||||
|  | ||||
|     def __unicode__(self): | ||||
|         return '<ActiveStatus: active={} last_active={} in_game={}>'.format(self.active, self.last_active, self.in_game) | ||||
|  | ||||
| class Enum(aenum.Enum): | ||||
|     """Used internally by fbchat to support enumerations""" | ||||
|     def __repr__(self): | ||||
|         # For documentation: | ||||
|         return '{}.{}'.format(type(self).__name__, self.name) | ||||
|  | ||||
| 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 | ||||
|  | ||||
| class ThreadLocation(Enum): | ||||
|     """Used to specify where a thread is located (inbox, pending, archived, other).""" | ||||
|     INBOX = 'INBOX' | ||||
|     PENDING = 'PENDING' | ||||
|     ARCHIVED = 'ARCHIVED' | ||||
|     OTHER = 'OTHER' | ||||
|  | ||||
| class TypingStatus(Enum): | ||||
|     """Used to specify whether the user is typing or has stopped typing""" | ||||
|     STOPPED = 0 | ||||
|     TYPING = 1 | ||||
|  | ||||
| class EmojiSize(Enum): | ||||
|     """Used to specify the size of a sent emoji""" | ||||
|     LARGE = '369239383222810' | ||||
|     MEDIUM = '369239343222814' | ||||
|     SMALL = '369239263222822' | ||||
|  | ||||
| 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' | ||||
|  | ||||
| class MessageReaction(Enum): | ||||
|     """Used to specify a message reaction""" | ||||
|     LOVE = '😍' | ||||
|     SMILE = '😆' | ||||
|     WOW = '😮' | ||||
|     SAD = '😢' | ||||
|     ANGRY = '😠' | ||||
|     YES = '👍' | ||||
|     NO = '👎' | ||||
| from ._core import Enum | ||||
| from ._exception import FBchatException, FBchatFacebookError, FBchatUserError | ||||
| from ._thread import ThreadType, ThreadLocation, ThreadColor, Thread | ||||
| from ._user import TypingStatus, User, ActiveStatus | ||||
| from ._group import Group, Room | ||||
| from ._page import Page | ||||
| from ._message import EmojiSize, MessageReaction, Mention, Message | ||||
| from ._attachment import Attachment, UnsentMessage, ShareAttachment | ||||
| from ._sticker import Sticker | ||||
| from ._location import LocationAttachment, LiveLocationAttachment | ||||
| from ._file import FileAttachment, AudioAttachment, ImageAttachment, VideoAttachment | ||||
| from ._quick_reply import ( | ||||
|     QuickReply, | ||||
|     QuickReplyText, | ||||
|     QuickReplyLocation, | ||||
|     QuickReplyPhoneNumber, | ||||
|     QuickReplyEmail, | ||||
| ) | ||||
| from ._poll import Poll, PollOption | ||||
| from ._plan import GuestStatus, Plan | ||||
|   | ||||
							
								
								
									
										317
									
								
								fbchat/utils.py
									
									
									
									
									
								
							
							
						
						
									
										317
									
								
								fbchat/utils.py
									
									
									
									
									
								
							| @@ -1,317 +0,0 @@ | ||||
| # -*- 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 | ||||
| import aenum | ||||
| from .models import * | ||||
|  | ||||
| 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" | ||||
| ] | ||||
|  | ||||
| LIKES = { | ||||
|     'large': EmojiSize.LARGE, | ||||
|     'medium': EmojiSize.MEDIUM, | ||||
|     'small': EmojiSize.SMALL, | ||||
|     'l': EmojiSize.LARGE, | ||||
|     'm': EmojiSize.MEDIUM, | ||||
|     's': EmojiSize.SMALL | ||||
| } | ||||
|  | ||||
|  | ||||
| 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 ReqUrl(object): | ||||
|     """A class containing all urls used by `fbchat`""" | ||||
|     SEARCH = "https://www.facebook.com/ajax/typeahead/search.php" | ||||
|     LOGIN = "https://m.facebook.com/login.php?login_attempt=1" | ||||
|     SEND = "https://www.facebook.com/messaging/send/" | ||||
|     UNREAD_THREADS = "https://www.facebook.com/ajax/mercury/unread_threads.php" | ||||
|     UNSEEN_THREADS = "https://www.facebook.com/mercury/unseen_thread_ids/" | ||||
|     THREADS = "https://www.facebook.com/ajax/mercury/threadlist_info.php" | ||||
|     MOVE_THREAD = "https://www.facebook.com/ajax/mercury/move_thread.php" | ||||
|     ARCHIVED_STATUS = "https://www.facebook.com/ajax/mercury/change_archived_status.php?dpr=1" | ||||
|     PINNED_STATUS = "https://www.facebook.com/ajax/mercury/change_pinned_status.php?dpr=1" | ||||
|     MESSAGES = "https://www.facebook.com/ajax/mercury/thread_info.php" | ||||
|     READ_STATUS = "https://www.facebook.com/ajax/mercury/change_read_status.php" | ||||
|     DELIVERED = "https://www.facebook.com/ajax/mercury/delivery_receipts.php" | ||||
|     MARK_SEEN = "https://www.facebook.com/ajax/mercury/mark_seen.php" | ||||
|     BASE = "https://www.facebook.com" | ||||
|     MOBILE = "https://m.facebook.com/" | ||||
|     STICKY = "https://0-edge-chat.facebook.com/pull" | ||||
|     PING = "https://0-edge-chat.facebook.com/active_ping" | ||||
|     UPLOAD = "https://upload.facebook.com/ajax/mercury/upload.php" | ||||
|     INFO = "https://www.facebook.com/chat/user_info/" | ||||
|     CONNECT = "https://www.facebook.com/ajax/add_friend/action.php?dpr=1" | ||||
|     REMOVE_USER = "https://www.facebook.com/chat/remove_participants/" | ||||
|     LOGOUT = "https://www.facebook.com/logout.php" | ||||
|     ALL_USERS = "https://www.facebook.com/chat/user_info_all" | ||||
|     SAVE_DEVICE = "https://m.facebook.com/login/save-device/cancel/" | ||||
|     CHECKPOINT = "https://m.facebook.com/login/checkpoint/" | ||||
|     THREAD_COLOR = "https://www.facebook.com/messaging/save_thread_color/?source=thread_settings&dpr=1" | ||||
|     THREAD_NICKNAME = "https://www.facebook.com/messaging/save_thread_nickname/?source=thread_settings&dpr=1" | ||||
|     THREAD_EMOJI = "https://www.facebook.com/messaging/save_thread_emoji/?source=thread_settings&dpr=1" | ||||
|     THREAD_IMAGE = "https://www.facebook.com/messaging/set_thread_image/?dpr=1" | ||||
|     THREAD_NAME = "https://www.facebook.com/messaging/set_thread_name/?dpr=1" | ||||
|     MESSAGE_REACTION = "https://www.facebook.com/webgraphql/mutation" | ||||
|     TYPING = "https://www.facebook.com/ajax/messaging/typ.php" | ||||
|     GRAPHQL = "https://www.facebook.com/api/graphqlbatch/" | ||||
|     ATTACHMENT_PHOTO = "https://www.facebook.com/mercury/attachments/photo/" | ||||
|     PLAN_CREATE = "https://www.facebook.com/ajax/eventreminder/create" | ||||
|     PLAN_INFO = "https://www.facebook.com/ajax/eventreminder" | ||||
|     PLAN_CHANGE = "https://www.facebook.com/ajax/eventreminder/submit" | ||||
|     PLAN_PARTICIPATION = "https://www.facebook.com/ajax/eventreminder/rsvp" | ||||
|     MODERN_SETTINGS_MENU = "https://www.facebook.com/bluebar/modern_settings_menu/" | ||||
|     REMOVE_FRIEND = "https://m.facebook.com/a/removefriend.php" | ||||
|     BLOCK_USER = "https://www.facebook.com/messaging/block_messages/?dpr=1" | ||||
|     UNBLOCK_USER = "https://www.facebook.com/messaging/unblock_messages/?dpr=1" | ||||
|     SAVE_ADMINS = "https://www.facebook.com/messaging/save_admins/?dpr=1" | ||||
|     APPROVAL_MODE = "https://www.facebook.com/messaging/set_approval_mode/?dpr=1" | ||||
|     CREATE_GROUP = "https://m.facebook.com/messages/send/?icm=1" | ||||
|     DELETE_THREAD = "https://www.facebook.com/ajax/mercury/delete_thread.php?dpr=1" | ||||
|     DELETE_MESSAGES = "https://www.facebook.com/ajax/mercury/delete_messages.php?dpr=1" | ||||
|     MUTE_THREAD = "https://www.facebook.com/ajax/mercury/change_mute_thread.php?dpr=1" | ||||
|     MUTE_REACTIONS = "https://www.facebook.com/ajax/mercury/change_reactions_mute_thread/?dpr=1" | ||||
|     MUTE_MENTIONS = "https://www.facebook.com/ajax/mercury/change_mentions_mute_thread/?dpr=1" | ||||
|     CREATE_POLL = "https://www.facebook.com/messaging/group_polling/create_poll/?dpr=1" | ||||
|     UPDATE_VOTE = "https://www.facebook.com/messaging/group_polling/update_vote/?dpr=1" | ||||
|     GET_POLL_OPTIONS = "https://www.facebook.com/ajax/mercury/get_poll_options" | ||||
|     SEARCH_MESSAGES = "https://www.facebook.com/ajax/mercury/search_snippets.php?dpr=1" | ||||
|     MARK_SPAM = "https://www.facebook.com/ajax/mercury/mark_spam.php?dpr=1" | ||||
|     UNSEND = "https://www.facebook.com/messaging/unsend_message/?dpr=1" | ||||
|  | ||||
|     pull_channel = 0 | ||||
|  | ||||
|     def change_pull_channel(self, channel=None): | ||||
|         if channel is None: | ||||
|             self.pull_channel = (self.pull_channel + 1) % 5 # Pull channel will be 0-4 | ||||
|         else: | ||||
|             self.pull_channel = channel | ||||
|         self.STICKY = "https://{}-edge-chat.facebook.com/pull".format(self.pull_channel) | ||||
|         self.PING = "https://{}-edge-chat.facebook.com/active_ping".format(self.pull_channel) | ||||
|  | ||||
|  | ||||
| facebookEncoding = 'UTF-8' | ||||
|  | ||||
| def now(): | ||||
|     return int(time()*1000) | ||||
|  | ||||
| def strip_to_json(text): | ||||
|     try: | ||||
|         return text[text.index('{'):] | ||||
|     except ValueError: | ||||
|         raise FBchatException('No JSON object found: {!r}'.format(text)) | ||||
|  | ||||
| def get_decoded_r(r): | ||||
|     return get_decoded(r._content) | ||||
|  | ||||
| def get_decoded(content): | ||||
|     return content.decode(facebookEncoding) | ||||
|  | ||||
| def parse_json(content): | ||||
|     return json.loads(content) | ||||
|  | ||||
| def get_json(r): | ||||
|     return json.loads(strip_to_json(get_decoded_r(r))) | ||||
|  | ||||
| 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 check_json(j): | ||||
|     if j.get('error') is None: | ||||
|         return | ||||
|     if 'errorDescription' in j: | ||||
|         # 'errorDescription' is in the users own language! | ||||
|         raise FBchatFacebookError('Error #{} when sending request: {}'.format(j['error'], j['errorDescription']), fb_error_code=j['error'], fb_error_message=j['errorDescription']) | ||||
|     elif 'debug_info' in j['error'] and 'code' in j['error']: | ||||
|         raise FBchatFacebookError('Error #{} when sending request: {}'.format(j['error']['code'], repr(j['error']['debug_info'])), fb_error_code=j['error']['code'], fb_error_message=j['error']['debug_info']) | ||||
|     else: | ||||
|         raise FBchatFacebookError('Error {} when sending request'.format(j['error']), fb_error_code=j['error']) | ||||
|  | ||||
| def check_request(r, as_json=True): | ||||
|     if not r.ok: | ||||
|         raise FBchatFacebookError('Error when sending request: Got {} response'.format(r.status_code), request_status_code=r.status_code) | ||||
|  | ||||
|     content = get_decoded_r(r) | ||||
|  | ||||
|     if content is None or len(content) == 0: | ||||
|         raise FBchatFacebookError('Error when sending request: Got empty response') | ||||
|  | ||||
|     if as_json: | ||||
|         content = strip_to_json(content) | ||||
|         try: | ||||
|             j = json.loads(content) | ||||
|         except ValueError: | ||||
|             raise FBchatFacebookError('Error while parsing JSON: {!r}'.format(content)) | ||||
|         check_json(j) | ||||
|         log.debug(j) | ||||
|         return j | ||||
|     else: | ||||
|         return content | ||||
|  | ||||
| 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 get_emojisize_from_tags(tags): | ||||
|     if tags is None: | ||||
|         return None | ||||
|     tmp = [tag for tag in tags if tag.startswith('hot_emoji_size:')] | ||||
|     if len(tmp) > 0: | ||||
|         try: | ||||
|             return LIKES[tmp[0].split(':')[1]] | ||||
|         except (KeyError, IndexError): | ||||
|             log.exception('Could not determine emoji size from {} - {}'.format(tags, tmp)) | ||||
|     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 | ||||
|         files.append(( | ||||
|             basename(file_url), | ||||
|             r.content, | ||||
|             r.headers.get('Content-Type') or guess_type(file_url)[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 enum_extend_if_invalid(enumeration, value): | ||||
|     try: | ||||
|         return enumeration(value) | ||||
|     except ValueError: | ||||
|         log.warning("Failed parsing {.__name__}({!r}). Extending enum.".format(enumeration, value)) | ||||
|         aenum.extend_enum(enumeration, "UNKNOWN_{}".format(value).upper(), value) | ||||
|         return enumeration(value) | ||||
|  | ||||
| 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] | ||||
| @@ -1,3 +1,6 @@ | ||||
| [tool.black] | ||||
| line-length = 88 | ||||
|  | ||||
| [build-system] | ||||
| requires = ["flit"] | ||||
| build-backend = "flit.buildapi" | ||||
| @@ -10,9 +13,11 @@ maintainer = "Mads Marquart" | ||||
| maintainer-email = "madsmtm@gmail.com" | ||||
| home-page = "https://github.com/carpedm20/fbchat/" | ||||
| requires = [ | ||||
|     "aenum", | ||||
|     "requests", | ||||
|     "beautifulsoup4", | ||||
|     "aenum~=2.0", | ||||
|     "attrs>=18.2", | ||||
|     "requests~=2.19", | ||||
|     "beautifulsoup4~=4.0", | ||||
|     "paho-mqtt~=1.5", | ||||
| ] | ||||
| description-file = "README.rst" | ||||
| classifiers = [ | ||||
| @@ -48,5 +53,16 @@ Repository = "https://github.com/carpedm20/fbchat/" | ||||
| [tool.flit.metadata.requires-extra] | ||||
| test = [ | ||||
|     "pytest~=4.0", | ||||
|     "six", | ||||
|     "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", | ||||
| ] | ||||
|   | ||||
| @@ -17,17 +17,21 @@ def user(client2): | ||||
|  | ||||
| @pytest.fixture(scope="session") | ||||
| def group(pytestconfig): | ||||
|     return {"id": load_variable("group_id", pytestconfig.cache), "type": ThreadType.GROUP} | ||||
|     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()]) | ||||
| ]) | ||||
| @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} | ||||
|         "none": {"id": "0", "type": ThreadType.GROUP}, | ||||
|     }[request.param] | ||||
|  | ||||
|  | ||||
| @@ -111,14 +115,14 @@ def compare(client, thread): | ||||
| def message_with_mentions(request, client, client2, group): | ||||
|     text = "Hi there [" | ||||
|     mentions = [] | ||||
|     if 'me' in request.param: | ||||
|     if "me" in request.param: | ||||
|         mentions.append(Mention(thread_id=client.uid, offset=len(text), length=2)) | ||||
|         text += "me, " | ||||
|     if 'other' in request.param: | ||||
|     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: | ||||
|     if "group" in request.param: | ||||
|         mentions.append(Mention(thread_id=group["id"], offset=len(text), length=5)) | ||||
|         text += "group, " | ||||
|     text += "nothing]" | ||||
|   | ||||
| @@ -27,7 +27,7 @@ def test_fetch_threads(client1): | ||||
| @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) | ||||
|     (message,) = client.fetchThreadMessages(limit=1) | ||||
|  | ||||
|     assert subset( | ||||
|         vars(message), uid=mid, author=client.uid, text=emoji, emoji_size=emoji_size | ||||
| @@ -46,9 +46,11 @@ def test_fetch_message_info_emoji(client, thread, emoji, emoji_size): | ||||
|  | ||||
| def test_fetch_message_mentions(client, thread, message_with_mentions): | ||||
|     mid = client.send(message_with_mentions) | ||||
|     message, = client.fetchThreadMessages(limit=1) | ||||
|     (message,) = client.fetchThreadMessages(limit=1) | ||||
|  | ||||
|     assert subset(vars(message), uid=mid, author=client.uid, text=message_with_mentions.text) | ||||
|     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] | ||||
| @@ -58,7 +60,9 @@ 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) | ||||
|     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] | ||||
| @@ -67,7 +71,7 @@ def test_fetch_message_info_mentions(client, thread, message_with_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) | ||||
|     (message,) = client.fetchThreadMessages(limit=1) | ||||
|  | ||||
|     assert subset(vars(message), uid=mid, author=client.uid) | ||||
|     assert subset(vars(message.sticker), uid=sticker.uid) | ||||
| @@ -92,6 +96,6 @@ def test_fetch_info(client1, group): | ||||
|  | ||||
| def test_fetch_image_url(client): | ||||
|     client.sendLocalFiles([path.join(path.dirname(__file__), "resources", "image.png")]) | ||||
|     message, = client.fetchThreadMessages(limit=1) | ||||
|     (message,) = client.fetchThreadMessages(limit=1) | ||||
|  | ||||
|     assert client.fetchImageUrl(message.attachments[0].uid) | ||||
|   | ||||
| @@ -19,5 +19,5 @@ def test_delete_messages(client): | ||||
|     mid1 = client.sendMessage(text1) | ||||
|     mid2 = client.sendMessage(text2) | ||||
|     client.deleteMessages(mid2) | ||||
|     message, = client.fetchThreadMessages(limit=1) | ||||
|     (message,) = client.fetchThreadMessages(limit=1) | ||||
|     assert subset(vars(message), uid=mid1, author=client.uid, text=text1) | ||||
|   | ||||
| @@ -9,17 +9,17 @@ from utils import random_hex, subset | ||||
| from time import time | ||||
|  | ||||
|  | ||||
| @pytest.fixture(scope="module", params=[ | ||||
| @pytest.fixture( | ||||
|     scope="module", | ||||
|     params=[ | ||||
|         Plan(int(time()) + 100, random_hex()), | ||||
|         pytest.param( | ||||
|             Plan(int(time()), random_hex()), | ||||
|         marks=[pytest.mark.xfail(raises=FBchatFacebookError)] | ||||
|             marks=[pytest.mark.xfail(raises=FBchatFacebookError)], | ||||
|         ), | ||||
|     pytest.param( | ||||
|         Plan(0, None), | ||||
|         marks=[pytest.mark.xfail()], | ||||
|     ), | ||||
| ]) | ||||
|         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"]) | ||||
| @@ -50,15 +50,14 @@ 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), | ||||
|         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): | ||||
| 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) | ||||
| @@ -94,7 +93,11 @@ 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"]) | ||||
|     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) | ||||
|   | ||||
| @@ -8,28 +8,40 @@ from fbchat.models import Poll, PollOption, ThreadType | ||||
| from utils import random_hex, subset | ||||
|  | ||||
|  | ||||
| @pytest.fixture(scope="module", params=[ | ||||
| @pytest.fixture( | ||||
|     scope="module", | ||||
|     params=[ | ||||
|         Poll(title=random_hex(), options=[]), | ||||
|     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=[ | ||||
|             ], | ||||
|         ), | ||||
|         Poll( | ||||
|             title=random_hex(), | ||||
|             options=[ | ||||
|                 PollOption(random_hex(), vote=False), | ||||
|                 PollOption(random_hex(), vote=False), | ||||
|     ]), | ||||
|     Poll(title=random_hex(), options=[ | ||||
|             ], | ||||
|         ), | ||||
|         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"]) | ||||
| @@ -45,11 +57,17 @@ def test_create_poll(client1, group, catch_event, poll_data): | ||||
|         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)) | ||||
|     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) | ||||
|         assert subset( | ||||
|             vars(recv_option), voters=voters, votes_count=len(voters), vote=False | ||||
|         ) | ||||
|  | ||||
|  | ||||
| def test_fetch_poll_options(client1, group, catch_event, poll_data): | ||||
| @@ -66,7 +84,11 @@ def test_update_poll_vote(client1, group, catch_event, poll_data): | ||||
|     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) | ||||
|         client1.updatePollVote( | ||||
|             event["poll"].uid, | ||||
|             option_ids=new_vote_ids + re_vote_ids, | ||||
|             new_options=new_options, | ||||
|         ) | ||||
|  | ||||
|     assert subset( | ||||
|         x.res, | ||||
| @@ -74,8 +96,12 @@ def test_update_poll_vote(client1, group, catch_event, poll_data): | ||||
|         thread_id=group["id"], | ||||
|         thread_type=ThreadType.GROUP, | ||||
|     ) | ||||
|     assert subset(vars(x.res["poll"]), title=poll.title, options_count=len(options + new_options)) | ||||
|     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) | ||||
|     assert set(x.res["removed_options"]) == set( | ||||
|         o.uid for o in options if o.vote and o.uid not in re_vote_ids | ||||
|     ) | ||||
|   | ||||
| @@ -38,7 +38,12 @@ def test_send_mentions(client, catch_event, compare, message_with_mentions): | ||||
|         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) | ||||
|     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] | ||||
| @@ -76,7 +81,15 @@ def test_send_images(client, catch_event, compare, method_name, url): | ||||
|  | ||||
|  | ||||
| def test_send_local_files(client, catch_event, compare): | ||||
|     files = ["image.png", "image.jpg", "image.gif", "file.json", "file.txt", "audio.mp3", "video.mp4"] | ||||
|     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( | ||||
| @@ -95,7 +108,10 @@ def test_send_remote_files(client, catch_event, compare): | ||||
|     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], | ||||
|             [ | ||||
|                 "https://github.com/carpedm20/fbchat/raw/master/tests/{}".format(f) | ||||
|                 for f in files | ||||
|             ], | ||||
|             message=Message(text), | ||||
|         ) | ||||
|  | ||||
| @@ -104,6 +120,6 @@ def test_send_remote_files(client, catch_event, compare): | ||||
|     assert len(x.res["message_object"].attachments) == len(files) | ||||
|  | ||||
|  | ||||
| @pytest.mark.parametrize('wave_first', [True, False]) | ||||
| @pytest.mark.parametrize("wave_first", [True, False]) | ||||
| def test_wave(client, wave_first): | ||||
|     client.wave(wave_first) | ||||
|   | ||||
| @@ -9,4 +9,4 @@ def test_catch_event(client2, catch_event): | ||||
|     mid = "test" | ||||
|     with catch_event("onMessage") as x: | ||||
|         client2.onMessage(mid=mid) | ||||
|     assert x.res['mid'] == mid | ||||
|     assert x.res["mid"] == mid | ||||
|   | ||||
| @@ -67,14 +67,19 @@ def test_change_nickname(client, client_all, catch_event, compare): | ||||
|     assert compare(x, changed_for=client_all.uid, new_nickname=nickname) | ||||
|  | ||||
|  | ||||
| @pytest.mark.parametrize("emoji", [ | ||||
| @pytest.mark.parametrize( | ||||
|     "emoji", | ||||
|     [ | ||||
|         "😀", | ||||
|         "😂", | ||||
|         "😕", | ||||
|         "😍", | ||||
|         pytest.param("🙃", marks=[pytest.mark.xfail(raises=FBchatFacebookError)]), | ||||
|     pytest.param("not an emoji", 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) | ||||
| @@ -85,7 +90,9 @@ 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"]) | ||||
|     assert subset( | ||||
|         x.res, new_image=image_id, author_id=client1.uid, thread_id=group["id"] | ||||
|     ) | ||||
|  | ||||
|  | ||||
| # To be changed when merged into master | ||||
| @@ -93,7 +100,9 @@ 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"]) | ||||
|     assert subset( | ||||
|         x.res, new_image=image_id, author_id=client1.uid, thread_id=group["id"] | ||||
|     ) | ||||
|  | ||||
|  | ||||
| @pytest.mark.parametrize( | ||||
| @@ -126,7 +135,7 @@ def test_typing_status(client, catch_event, compare, status): | ||||
|     assert compare(x, status=status) | ||||
|  | ||||
|  | ||||
| @pytest.mark.parametrize('require_admin_approval', [True, False]) | ||||
| @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"]) | ||||
| @@ -138,6 +147,7 @@ def test_change_approval_mode(client1, group, catch_event, require_admin_approva | ||||
|         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) | ||||
|   | ||||
| @@ -106,7 +106,7 @@ 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', | ||||
|         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, | ||||
|     ) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user