Compare commits
	
		
			72 Commits
		
	
	
		
			v1.7.3
			...
			064707ac23
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 064707ac23 | ||
|  | b9b4d57b25 | ||
|  | b4618739f3 | ||
|  | 22c6c82c0e | ||
|  | 19c875c18a | ||
|  | 12bbc0058c | ||
|  | 9c81806b95 | ||
|  | 45303005b8 | ||
|  | 881aa9adce | ||
|  | 4714be5697 | ||
|  | cb7f4a72d7 | ||
|  | fb63ff0db8 | ||
|  | c5f447e20b | ||
|  | b4d3769fd5 | ||
|  | b199d597b2 | ||
|  | debfb37a47 | ||
|  | 67fd6ffdf6 | ||
|  | e57265016e | ||
|  | cf4c22898c | ||
|  | 3bb99541e7 | ||
|  | 8c367af0ff | ||
|  | bc1e3edf17 | ||
|  | e488f4a7da | ||
|  | afad38d8e1 | ||
|  | e9804d4184 | ||
|  | a1b80a7abb | ||
|  | 803bfa7084 | ||
|  | d1cb866b44 | ||
|  | a298e0cf16 | ||
|  | 766b0125fb | ||
|  | 998fa43fb2 | ||
|  | ecc6edac5a | ||
|  | ea518ba4c9 | ||
|  | ffdf4222bf | ||
|  | a97ef67411 | ||
|  | 813219cd9c | ||
|  | bb1f7d9294 | ||
|  | 3d28c958d3 | ||
|  | 6b68916d74 | ||
|  | 12e752e681 | ||
|  | 1f342d0c71 | ||
|  | 5e86d4a48a | ||
|  | 0838f84859 | ||
|  | abc938eacd | ||
|  | 4d13cd2c0b | ||
|  | 8f8971c706 | ||
|  | 2703d9513a | ||
|  | 3dce83de93 | ||
|  | ef8e7d4251 | ||
|  | a131e1ae73 | ||
|  | 84a86bd7bd | ||
|  | adfb5886c9 | ||
|  | 8d237ea4ef | ||
|  | 513bc6eadf | ||
|  | 856962af63 | ||
|  | 7c68a29181 | ||
|  | 2f4e3f2bb1 | ||
|  | 0389b838bc | ||
|  | 441f53e382 | ||
|  | 83c45dcf40 | ||
|  | cc9d81a39e | ||
|  | edf14cfd84 | ||
|  | ee79969eda | ||
|  | dbb20b1fdc | ||
|  | beee209249 | ||
|  | d6876ce13b | ||
|  | ed05d16a31 | ||
|  | 3806f01d2f | ||
|  | 5b69ced1e8 | ||
|  | 6b07f1d8b9 | ||
|  | 700cf14a50 | ||
|  | 1b08243cd2 | 
| @@ -1,5 +1,5 @@ | ||||
| [bumpversion] | ||||
| current_version = 1.7.3 | ||||
| current_version = 1.9.6 | ||||
| commit = True | ||||
| tag = True | ||||
|  | ||||
|   | ||||
							
								
								
									
										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 | ||||
|   | ||||
| @@ -30,7 +30,7 @@ jobs: | ||||
|     script: black --check --verbose . | ||||
|  | ||||
|   - stage: deploy | ||||
|     name: Github Releases | ||||
|     name: GitHub Releases | ||||
|     if: tag IS present | ||||
|     install: skip | ||||
|     script: flit build | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| Contributing to fbchat | ||||
| ====================== | ||||
| Contributing to ``fbchat`` | ||||
| ========================== | ||||
|  | ||||
| Thanks for reading this, all contributions are very much welcome! | ||||
|  | ||||
|   | ||||
| @@ -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 | ||||
|   | ||||
							
								
								
									
										28
									
								
								docs/conf.py
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								docs/conf.py
									
									
									
									
									
								
							| @@ -41,6 +41,8 @@ extensions = [ | ||||
|     "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. | ||||
| @@ -93,8 +95,6 @@ html_theme_options = { | ||||
|     "show_related": False, | ||||
| } | ||||
|  | ||||
| html_extra_path = ["robots.txt"] | ||||
|  | ||||
| # Custom sidebar templates, must be a dictionary that maps document names | ||||
| # to template names. | ||||
| # | ||||
| @@ -182,3 +182,27 @@ intersphinx_mapping = {"https://docs.python.org/": None} | ||||
|  | ||||
| # If true, `todo` and `todoList` produce output, else they produce nothing. | ||||
| todo_include_todos = True | ||||
|  | ||||
| todo_link_only = True | ||||
|  | ||||
| # -- Options for napoleon extension ---------------------------------------------- | ||||
|  | ||||
| # Use Google style docstrings | ||||
| napoleon_google_docstring = True | ||||
| napoleon_numpy_docstring = False | ||||
|  | ||||
| # napoleon_use_admonition_for_examples = False | ||||
| # napoleon_use_admonition_for_notes = False | ||||
| # napoleon_use_admonition_for_references = False | ||||
|  | ||||
| # -- Options for spelling extension ---------------------------------------------- | ||||
|  | ||||
| spelling_word_list_filename = [ | ||||
|     "spelling/names.txt", | ||||
|     "spelling/technical.txt", | ||||
|     "spelling/fixes.txt", | ||||
| ] | ||||
| spelling_ignore_wiki_words = False | ||||
| # spelling_ignore_acronyms = False | ||||
| spelling_ignore_python_builtins = False | ||||
| spelling_ignore_importable_modules = False | ||||
|   | ||||
| @@ -30,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 | ||||
|  | ||||
|   | ||||
| @@ -9,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 | ||||
|  | ||||
| @@ -28,7 +28,7 @@ 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 | ||||
|   | ||||
| @@ -6,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`) | ||||
|  | ||||
| @@ -35,7 +35,7 @@ This means doing the exact same GET/POST requests and tricking Facebook into thi | ||||
| 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, | ||||
| @@ -44,7 +44,7 @@ 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`` | ||||
|  | ||||
|   | ||||
| @@ -3,10 +3,10 @@ | ||||
| Installation | ||||
| ============ | ||||
|  | ||||
| Pip Install fbchat | ||||
| ------------------ | ||||
| Install using pip | ||||
| ----------------- | ||||
|  | ||||
| To install fbchat, run this command: | ||||
| To install ``fbchat``, run this command: | ||||
|  | ||||
| .. code-block:: sh | ||||
|  | ||||
| @@ -19,7 +19,7 @@ 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: | ||||
|   | ||||
| @@ -24,7 +24,7 @@ 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`) | ||||
|  | ||||
| @@ -125,8 +125,8 @@ 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 | ||||
|  | ||||
| @@ -154,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: | ||||
| @@ -192,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 | ||||
|   | ||||
| @@ -1,2 +0,0 @@ | ||||
| User-agent: * | ||||
| Disallow: /en/master/ | ||||
							
								
								
									
										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 | ||||
| @@ -9,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 | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| # -*- coding: UTF-8 -*- | ||||
|  | ||||
| from itertools import islice | ||||
| from fbchat import Client | ||||
| from fbchat.models import * | ||||
|  | ||||
| @@ -62,3 +63,9 @@ 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) | ||||
|   | ||||
| @@ -47,7 +47,7 @@ client.sendLocalImage( | ||||
|     thread_type=thread_type, | ||||
| ) | ||||
|  | ||||
| # Will download the image at the url `<image url>`, and then send it | ||||
| # 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"), | ||||
|   | ||||
| @@ -13,7 +13,7 @@ from ._client import Client | ||||
| from ._util import log  # TODO: Remove this (from examples too) | ||||
|  | ||||
| __title__ = "fbchat" | ||||
| __version__ = "1.7.3" | ||||
| __version__ = "1.9.6" | ||||
| __description__ = "Facebook Chat (Messenger) for Python" | ||||
|  | ||||
| __copyright__ = "Copyright 2015 - 2019 by Taehoon Kim" | ||||
|   | ||||
| @@ -7,7 +7,7 @@ from . import _util | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class Attachment(object): | ||||
|     """Represents a Facebook attachment""" | ||||
|     """Represents a Facebook attachment.""" | ||||
|  | ||||
|     #: The attachment ID | ||||
|     uid = attr.ib(None) | ||||
| @@ -15,12 +15,12 @@ class Attachment(object): | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class UnsentMessage(Attachment): | ||||
|     """Represents an unsent message attachment""" | ||||
|     """Represents an unsent message attachment.""" | ||||
|  | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class ShareAttachment(Attachment): | ||||
|     """Represents a shared item (eg. URL) that has been sent as a Facebook attachment""" | ||||
|     """Represents a shared item (e.g. URL) attachment.""" | ||||
|  | ||||
|     #: ID of the author of the shared post | ||||
|     author = attr.ib(None) | ||||
|   | ||||
							
								
								
									
										2721
									
								
								fbchat/_client.py
									
									
									
									
									
								
							
							
						
						
									
										2721
									
								
								fbchat/_client.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -8,7 +8,7 @@ log = logging.getLogger("client") | ||||
|  | ||||
|  | ||||
| class Enum(aenum.Enum): | ||||
|     """Used internally by fbchat to support enumerations""" | ||||
|     """Used internally by ``fbchat`` to support enumerations""" | ||||
|  | ||||
|     def __repr__(self): | ||||
|         # For documentation: | ||||
|   | ||||
| @@ -3,7 +3,10 @@ from __future__ import unicode_literals | ||||
|  | ||||
|  | ||||
| class FBchatException(Exception): | ||||
|     """Custom exception thrown by fbchat. All exceptions in the fbchat module inherits this""" | ||||
|     """Custom exception thrown by ``fbchat``. | ||||
|  | ||||
|     All exceptions in the ``fbchat`` module inherits this. | ||||
|     """ | ||||
|  | ||||
|  | ||||
| class FBchatFacebookError(FBchatException): | ||||
| @@ -11,7 +14,7 @@ class FBchatFacebookError(FBchatException): | ||||
|     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) | ||||
|     #: 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__( | ||||
| @@ -22,7 +25,7 @@ class FBchatFacebookError(FBchatException): | ||||
|         request_status_code=None, | ||||
|     ): | ||||
|         super(FBchatFacebookError, self).__init__(message) | ||||
|         """Thrown by fbchat when Facebook returns an error""" | ||||
|         """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 | ||||
| @@ -54,4 +57,4 @@ class FBchatPleaseRefresh(FBchatFacebookError): | ||||
|  | ||||
|  | ||||
| class FBchatUserError(FBchatException): | ||||
|     """Thrown by fbchat when wrong values are entered""" | ||||
|     """Thrown by ``fbchat`` when wrong values are entered.""" | ||||
|   | ||||
| @@ -7,9 +7,9 @@ from ._attachment import Attachment | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class FileAttachment(Attachment): | ||||
|     """Represents a file that has been sent as a Facebook attachment""" | ||||
|     """Represents a file that has been sent as a Facebook attachment.""" | ||||
|  | ||||
|     #: Url where you can download the file | ||||
|     #: URL where you can download the file | ||||
|     url = attr.ib(None) | ||||
|     #: Size of the file in bytes | ||||
|     size = attr.ib(None) | ||||
| @@ -33,13 +33,13 @@ class FileAttachment(Attachment): | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class AudioAttachment(Attachment): | ||||
|     """Represents an audio file that has been sent as a Facebook 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 of the audio file | ||||
|     url = attr.ib(None) | ||||
|     #: Duration of the audioclip in milliseconds | ||||
|     #: Duration of the audio clip in milliseconds | ||||
|     duration = attr.ib(None) | ||||
|     #: Audio type | ||||
|     audio_type = attr.ib(None) | ||||
| @@ -59,13 +59,13 @@ class AudioAttachment(Attachment): | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class ImageAttachment(Attachment): | ||||
|     """Represents an image that has been sent as a Facebook attachment | ||||
|     """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 | ||||
|     To retrieve the full image URL, use: `Client.fetchImageUrl`, and pass it the id of | ||||
|     the image attachment. | ||||
|     """ | ||||
|  | ||||
|     #: The extension of the original image (eg. 'png') | ||||
|     #: 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)) | ||||
| @@ -92,7 +92,7 @@ class ImageAttachment(Attachment): | ||||
|     #: Height of the large preview image | ||||
|     large_preview_height = attr.ib(None) | ||||
|  | ||||
|     #: URL to an animated preview of the image (eg. for gifs) | ||||
|     #: 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) | ||||
| @@ -155,10 +155,22 @@ class ImageAttachment(Attachment): | ||||
|             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""" | ||||
|     """Represents a video that has been sent as a Facebook attachment.""" | ||||
|  | ||||
|     #: Size of the original video in bytes | ||||
|     size = attr.ib(None) | ||||
| @@ -252,6 +264,18 @@ class VideoAttachment(Attachment): | ||||
|             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"] | ||||
|   | ||||
| @@ -8,11 +8,11 @@ from ._thread import ThreadType, Thread | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class Group(Thread): | ||||
|     """Represents a Facebook group. Inherits `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 dict, containing user nicknames mapped to their IDs | ||||
|     #: 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) | ||||
| @@ -104,10 +104,13 @@ class Group(Thread): | ||||
|             plan=plan, | ||||
|         ) | ||||
|  | ||||
|     def _to_send_data(self): | ||||
|         return {"thread_fbid": self.uid} | ||||
|  | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class Room(Group): | ||||
|     """Deprecated. Use :class:`Group` instead""" | ||||
|     """Deprecated. Use `Group` instead.""" | ||||
|  | ||||
|     # True is room is not discoverable | ||||
|     privacy_mode = attr.ib(None) | ||||
|   | ||||
| @@ -8,9 +8,9 @@ from . import _util | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class LocationAttachment(Attachment): | ||||
|     """Represents a user location | ||||
|     """Represents a user location. | ||||
|  | ||||
|     Latitude and longitude OR address is provided by Facebook | ||||
|     Latitude and longitude OR address is provided by Facebook. | ||||
|     """ | ||||
|  | ||||
|     #: Latitude of the location | ||||
| @@ -58,7 +58,7 @@ class LocationAttachment(Attachment): | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class LiveLocationAttachment(LocationAttachment): | ||||
|     """Represents a live user location""" | ||||
|     """Represents a live user location.""" | ||||
|  | ||||
|     #: Name of the location | ||||
|     name = attr.ib(None) | ||||
|   | ||||
| @@ -9,7 +9,7 @@ from ._core import Enum | ||||
|  | ||||
|  | ||||
| class EmojiSize(Enum): | ||||
|     """Used to specify the size of a sent emoji""" | ||||
|     """Used to specify the size of a sent emoji.""" | ||||
|  | ||||
|     LARGE = "369239383222810" | ||||
|     MEDIUM = "369239343222814" | ||||
| @@ -26,14 +26,14 @@ class EmojiSize(Enum): | ||||
|             "s": cls.SMALL, | ||||
|         } | ||||
|         for tag in tags or (): | ||||
|             data = tag.split(":", maxsplit=1) | ||||
|             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""" | ||||
|     """Used to specify a message reaction.""" | ||||
|  | ||||
|     HEART = "❤" | ||||
|     LOVE = "😍" | ||||
| @@ -47,7 +47,7 @@ class MessageReaction(Enum): | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class Mention(object): | ||||
|     """Represents a @mention""" | ||||
|     """Represents a ``@mention``.""" | ||||
|  | ||||
|     #: The thread ID the mention is pointing at | ||||
|     thread_id = attr.ib() | ||||
| @@ -59,7 +59,7 @@ class Mention(object): | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class Message(object): | ||||
|     """Represents a Facebook message""" | ||||
|     """Represents a Facebook message.""" | ||||
|  | ||||
|     #: The actual message | ||||
|     text = attr.ib(None) | ||||
| @@ -75,9 +75,9 @@ class Message(object): | ||||
|     timestamp = attr.ib(None, init=False) | ||||
|     #: Whether the message is read | ||||
|     is_read = attr.ib(None, init=False) | ||||
|     #: A list of pepole IDs who read the message, works only with :func:`fbchat.Client.fetchThreadMessages` | ||||
|     #: 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 dict with user's IDs as keys, and their :class:`MessageReaction` as values | ||||
|     #: 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) | ||||
| @@ -98,7 +98,7 @@ class Message(object): | ||||
|     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. | ||||
|         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=[]> | ||||
| @@ -151,6 +151,55 @@ class Message(object): | ||||
|             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: | ||||
|   | ||||
							
								
								
									
										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) | ||||
| @@ -8,9 +8,9 @@ from ._thread import ThreadType, Thread | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class Page(Thread): | ||||
|     """Represents a Facebook page. Inherits `Thread`""" | ||||
|     """Represents a Facebook page. Inherits `Thread`.""" | ||||
|  | ||||
|     #: The page's custom url | ||||
|     #: The page's custom URL | ||||
|     url = attr.ib(None) | ||||
|     #: The name of the page's location city | ||||
|     city = attr.ib(None) | ||||
|   | ||||
| @@ -14,11 +14,11 @@ class GuestStatus(Enum): | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class Plan(object): | ||||
|     """Represents a plan""" | ||||
|     """Represents a plan.""" | ||||
|  | ||||
|     #: ID of the plan | ||||
|     uid = attr.ib(None, init=False) | ||||
|     #: Plan time (unix time stamp), only precise down to the minute | ||||
|     #: Plan time (timestamp), only precise down to the minute | ||||
|     time = attr.ib(converter=int) | ||||
|     #: Plan title | ||||
|     title = attr.ib() | ||||
| @@ -28,7 +28,7 @@ class Plan(object): | ||||
|     location_id = attr.ib(None, converter=lambda x: x or "") | ||||
|     #: ID of the plan creator | ||||
|     author_id = attr.ib(None, init=False) | ||||
|     #: Dict of `User` IDs mapped to their `GuestStatus` | ||||
|     #: Dictionary of `User` IDs mapped to their `GuestStatus` | ||||
|     guests = attr.ib(None, init=False) | ||||
|  | ||||
|     @property | ||||
|   | ||||
| @@ -6,7 +6,7 @@ import attr | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class Poll(object): | ||||
|     """Represents a poll""" | ||||
|     """Represents a poll.""" | ||||
|  | ||||
|     #: Title of the poll | ||||
|     title = attr.ib() | ||||
| @@ -29,7 +29,7 @@ class Poll(object): | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class PollOption(object): | ||||
|     """Represents a poll option""" | ||||
|     """Represents a poll option.""" | ||||
|  | ||||
|     #: Text of the poll option | ||||
|     text = attr.ib() | ||||
|   | ||||
| @@ -7,7 +7,7 @@ from ._attachment import Attachment | ||||
|  | ||||
| @attr.s(cmp=False) | ||||
| class QuickReply(object): | ||||
|     """Represents a quick reply""" | ||||
|     """Represents a quick reply.""" | ||||
|  | ||||
|     #: Payload of the quick reply | ||||
|     payload = attr.ib(None) | ||||
| @@ -21,7 +21,7 @@ class QuickReply(object): | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class QuickReplyText(QuickReply): | ||||
|     """Represents a text quick reply""" | ||||
|     """Represents a text quick reply.""" | ||||
|  | ||||
|     #: Title of the quick reply | ||||
|     title = attr.ib(None) | ||||
| @@ -38,7 +38,7 @@ class QuickReplyText(QuickReply): | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class QuickReplyLocation(QuickReply): | ||||
|     """Represents a location quick reply (Doesn't work on mobile)""" | ||||
|     """Represents a location quick reply (Doesn't work on mobile).""" | ||||
|  | ||||
|     #: Type of the quick reply | ||||
|     _type = "location" | ||||
| @@ -50,7 +50,7 @@ class QuickReplyLocation(QuickReply): | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class QuickReplyPhoneNumber(QuickReply): | ||||
|     """Represents a phone number quick reply (Doesn't work on mobile)""" | ||||
|     """Represents a phone number quick reply (Doesn't work on mobile).""" | ||||
|  | ||||
|     #: URL of the quick reply image (optional) | ||||
|     image_url = attr.ib(None) | ||||
| @@ -64,7 +64,7 @@ class QuickReplyPhoneNumber(QuickReply): | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class QuickReplyEmail(QuickReply): | ||||
|     """Represents an email quick reply (Doesn't work on mobile)""" | ||||
|     """Represents an email quick reply (Doesn't work on mobile).""" | ||||
|  | ||||
|     #: URL of the quick reply image (optional) | ||||
|     image_url = attr.ib(None) | ||||
|   | ||||
							
								
								
									
										158
									
								
								fbchat/_state.py
									
									
									
									
									
								
							
							
						
						
									
										158
									
								
								fbchat/_state.py
									
									
									
									
									
								
							| @@ -7,11 +7,19 @@ import re | ||||
| import requests | ||||
| import random | ||||
|  | ||||
| from . import _util, _exception | ||||
| 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")) | ||||
|  | ||||
| @@ -24,6 +32,10 @@ def session_factory(user_agent=None): | ||||
|     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 `/` | ||||
| @@ -91,25 +103,21 @@ def _2fa_helper(session, code, r): | ||||
| class State(object): | ||||
|     """Stores and manages state required for most Facebook requests.""" | ||||
|  | ||||
|     fb_dtsg = attr.ib() | ||||
|     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_user_id(self): | ||||
|         rtn = self.get_cookies().get("c_user") | ||||
|         if rtn is None: | ||||
|             return None | ||||
|         return str(rtn) | ||||
|  | ||||
|     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, | ||||
|             "fb_dtsg": self._fb_dtsg, | ||||
|         } | ||||
|  | ||||
|     @classmethod | ||||
| @@ -163,6 +171,9 @@ class State(object): | ||||
|  | ||||
|     @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) | ||||
| @@ -180,7 +191,11 @@ class State(object): | ||||
|         logout_h = logout_h_element["value"] if logout_h_element else None | ||||
|  | ||||
|         return cls( | ||||
|             fb_dtsg=fb_dtsg, revision=revision, session=session, logout_h=logout_h | ||||
|             user_id=user_id, | ||||
|             fb_dtsg=fb_dtsg, | ||||
|             revision=revision, | ||||
|             session=session, | ||||
|             logout_h=logout_h, | ||||
|         ) | ||||
|  | ||||
|     def get_cookies(self): | ||||
| @@ -191,3 +206,126 @@ class State(object): | ||||
|         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) | ||||
|             ) | ||||
|   | ||||
| @@ -7,7 +7,7 @@ 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""" | ||||
|     """Represents a Facebook sticker that has been sent to a thread as an attachment.""" | ||||
|  | ||||
|     #: The sticker-pack's ID | ||||
|     pack = attr.ib(None) | ||||
| @@ -21,7 +21,7 @@ class Sticker(Attachment): | ||||
|     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. coloumn | ||||
|     #: 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) | ||||
|   | ||||
| @@ -6,13 +6,27 @@ 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""" | ||||
|     """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).""" | ||||
| @@ -24,7 +38,7 @@ class ThreadLocation(Enum): | ||||
|  | ||||
|  | ||||
| class ThreadColor(Enum): | ||||
|     """Used to specify a thread colors""" | ||||
|     """Used to specify a thread colors.""" | ||||
|  | ||||
|     MESSENGER_BLUE = "#0084ff" | ||||
|     VIKING = "#44bec7" | ||||
| @@ -60,13 +74,13 @@ class ThreadColor(Enum): | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class Thread(object): | ||||
|     """Represents a Facebook thread""" | ||||
|     """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 | ||||
|     #: A URL to the thread's picture | ||||
|     photo = attr.ib(None) | ||||
|     #: The name of the thread | ||||
|     name = attr.ib(None) | ||||
| @@ -127,3 +141,7 @@ class Thread(object): | ||||
|                 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} | ||||
|   | ||||
| @@ -38,7 +38,7 @@ GENDERS = { | ||||
|  | ||||
|  | ||||
| class TypingStatus(Enum): | ||||
|     """Used to specify whether the user is typing or has stopped typing""" | ||||
|     """Used to specify whether the user is typing or has stopped typing.""" | ||||
|  | ||||
|     STOPPED = 0 | ||||
|     TYPING = 1 | ||||
| @@ -46,9 +46,9 @@ class TypingStatus(Enum): | ||||
|  | ||||
| @attr.s(cmp=False, init=False) | ||||
| class User(Thread): | ||||
|     """Represents a Facebook user. Inherits `Thread`""" | ||||
|     """Represents a Facebook user. Inherits `Thread`.""" | ||||
|  | ||||
|     #: The profile url | ||||
|     #: The profile URL | ||||
|     url = attr.ib(None) | ||||
|     #: The users first name | ||||
|     first_name = attr.ib(None) | ||||
| @@ -192,17 +192,6 @@ class ActiveStatus(object): | ||||
|     in_game = attr.ib(None) | ||||
|  | ||||
|     @classmethod | ||||
|     def _from_chatproxy_presence(cls, id_, data): | ||||
|         return cls( | ||||
|             active=data["p"] in [2, 3] if "p" in data else None, | ||||
|             last_active=data.get("lat"), | ||||
|             in_game=int(id_) in data.get("gamers", {}), | ||||
|         ) | ||||
|  | ||||
|     @classmethod | ||||
|     def _from_buddylist_overlay(cls, data, in_game=None): | ||||
|         return cls( | ||||
|             active=data["a"] in [2, 3] if "a" in data else None, | ||||
|             last_active=data.get("la"), | ||||
|             in_game=None, | ||||
|         ) | ||||
|     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) | ||||
|   | ||||
| @@ -57,6 +57,11 @@ 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: | ||||
| @@ -65,6 +70,14 @@ def strip_json_cruft(text): | ||||
|         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) | ||||
|  | ||||
| @@ -219,11 +232,12 @@ def get_files_from_urls(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( | ||||
|             ( | ||||
|                 basename(file_url).split("?")[0].split("#")[0], | ||||
|                 file_name, | ||||
|                 r.content, | ||||
|                 r.headers.get("Content-Type") or guess_type(file_url)[0], | ||||
|                 r.headers.get("Content-Type") or guess_type(file_name)[0], | ||||
|             ) | ||||
|         ) | ||||
|     return files | ||||
|   | ||||
| @@ -17,6 +17,7 @@ requires = [ | ||||
|     "attrs>=18.2", | ||||
|     "requests~=2.19", | ||||
|     "beautifulsoup4~=4.0", | ||||
|     "paho-mqtt~=1.5", | ||||
| ] | ||||
| description-file = "README.rst" | ||||
| classifiers = [ | ||||
| @@ -56,6 +57,7 @@ test = [ | ||||
| ] | ||||
| docs = [ | ||||
|     "sphinx~=2.0", | ||||
|     "sphinxcontrib-spelling~=4.0" | ||||
| ] | ||||
| lint = [ | ||||
|     "black", | ||||
|   | ||||
| @@ -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,7 +46,7 @@ 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 | ||||
| @@ -71,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) | ||||
| @@ -96,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) | ||||
|   | ||||
| @@ -63,7 +63,7 @@ def test_create_poll(client1, group, catch_event, poll_data): | ||||
|     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)) | ||||
|         (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 | ||||
|   | ||||
		Reference in New Issue
	
	Block a user