Merge pull request #436 from carpedm20/clean-up-requests
Normalize a lot of request parsing
This commit is contained in:
		| @@ -118,81 +118,42 @@ class Client(object): | |||||||
|             return True |             return True | ||||||
|         return False |         return False | ||||||
|  |  | ||||||
|     def _get(self, url, query=None, fix_request=False, as_json=False, error_retries=3): |     def _get(self, url, query=None, error_retries=3): | ||||||
|         payload = self._generatePayload(query) |         payload = self._generatePayload(query) | ||||||
|         r = self._state._session.get(prefix_url(url), params=payload) |         r = self._state._session.get(prefix_url(url), params=payload) | ||||||
|         if not fix_request: |  | ||||||
|             return r |  | ||||||
|         try: |         try: | ||||||
|             return check_request(r, as_json=as_json) |             return check_request(r) | ||||||
|         except FBchatFacebookError as e: |         except FBchatFacebookError as e: | ||||||
|             if error_retries > 0 and self._fix_fb_errors(e.fb_error_code): |             if error_retries > 0 and self._fix_fb_errors(e.fb_error_code): | ||||||
|                 return self._get( |                 return self._get(url, query=query, error_retries=error_retries - 1) | ||||||
|                     url, |  | ||||||
|                     query=query, |  | ||||||
|                     fix_request=fix_request, |  | ||||||
|                     as_json=as_json, |  | ||||||
|                     error_retries=error_retries - 1, |  | ||||||
|                 ) |  | ||||||
|             raise e |             raise e | ||||||
|  |  | ||||||
|     def _post( |     def _post(self, url, query=None, files=None, as_graphql=False, error_retries=3): | ||||||
|         self, |  | ||||||
|         url, |  | ||||||
|         query=None, |  | ||||||
|         fix_request=False, |  | ||||||
|         as_json=False, |  | ||||||
|         as_graphql=False, |  | ||||||
|         error_retries=3, |  | ||||||
|     ): |  | ||||||
|         payload = self._generatePayload(query) |         payload = self._generatePayload(query) | ||||||
|         r = self._state._session.post(prefix_url(url), data=payload) |         r = self._state._session.post(prefix_url(url), data=payload, files=files) | ||||||
|         if not fix_request: |  | ||||||
|             return r |  | ||||||
|         try: |         try: | ||||||
|             if as_graphql: |             if as_graphql: | ||||||
|                 content = check_request(r, as_json=False) |                 content = check_request(r, as_json=False) | ||||||
|                 return graphql_response_to_json(content) |                 return graphql_response_to_json(content) | ||||||
|             else: |             else: | ||||||
|                 return check_request(r, as_json=as_json) |                 return check_request(r) | ||||||
|         except FBchatFacebookError as e: |         except FBchatFacebookError as e: | ||||||
|             if error_retries > 0 and self._fix_fb_errors(e.fb_error_code): |             if error_retries > 0 and self._fix_fb_errors(e.fb_error_code): | ||||||
|                 return self._post( |                 return self._post( | ||||||
|                     url, |                     url, | ||||||
|                     query=query, |                     query=query, | ||||||
|                     fix_request=fix_request, |                     files=files, | ||||||
|                     as_json=as_json, |  | ||||||
|                     as_graphql=as_graphql, |                     as_graphql=as_graphql, | ||||||
|                     error_retries=error_retries - 1, |                     error_retries=error_retries - 1, | ||||||
|                 ) |                 ) | ||||||
|             raise e |             raise e | ||||||
|  |  | ||||||
|     def _postFile( |     def _payload_post(self, url, data, files=None): | ||||||
|         self, |         j = self._post(url, data, files=files) | ||||||
|         url, |  | ||||||
|         files=None, |  | ||||||
|         query=None, |  | ||||||
|         fix_request=False, |  | ||||||
|         as_json=False, |  | ||||||
|         error_retries=3, |  | ||||||
|     ): |  | ||||||
|         payload = self._generatePayload(query) |  | ||||||
|         r = self._state._session.post(prefix_url(url), data=payload, files=files) |  | ||||||
|         if not fix_request: |  | ||||||
|             return r |  | ||||||
|         try: |         try: | ||||||
|             return check_request(r, as_json=as_json) |             return j["payload"] | ||||||
|         except FBchatFacebookError as e: |         except (KeyError, TypeError): | ||||||
|             if error_retries > 0 and self._fix_fb_errors(e.fb_error_code): |             raise FBchatException("Missing payload: {}".format(j)) | ||||||
|                 return self._postFile( |  | ||||||
|                     url, |  | ||||||
|                     files=files, |  | ||||||
|                     query=query, |  | ||||||
|                     fix_request=fix_request, |  | ||||||
|                     as_json=as_json, |  | ||||||
|                     error_retries=error_retries - 1, |  | ||||||
|                 ) |  | ||||||
|             raise e |  | ||||||
|  |  | ||||||
|     def graphql_requests(self, *queries): |     def graphql_requests(self, *queries): | ||||||
|         """ |         """ | ||||||
| @@ -208,9 +169,7 @@ class Client(object): | |||||||
|             "response_format": "json", |             "response_format": "json", | ||||||
|             "queries": graphql_queries_to_json(*queries), |             "queries": graphql_queries_to_json(*queries), | ||||||
|         } |         } | ||||||
|         return tuple( |         return tuple(self._post("/api/graphqlbatch/", data, as_graphql=True)) | ||||||
|             self._post("/api/graphqlbatch/", data, fix_request=True, as_graphql=True) |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def graphql_request(self, query): |     def graphql_request(self, query): | ||||||
|         """ |         """ | ||||||
| @@ -454,14 +413,10 @@ class Client(object): | |||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         data = {"viewer": self._uid} |         data = {"viewer": self._uid} | ||||||
|         j = self._post( |         j = self._payload_post("/chat/user_info_all", data) | ||||||
|             "/chat/user_info_all", query=data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|         if j.get("payload") is None: |  | ||||||
|             raise FBchatException("Missing payload while fetching users: {}".format(j)) |  | ||||||
|  |  | ||||||
|         users = [] |         users = [] | ||||||
|         for data in j["payload"].values(): |         for data in j.values(): | ||||||
|             if data["type"] in ["user", "friend"]: |             if data["type"] in ["user", "friend"]: | ||||||
|                 if data["id"] in ["0", 0]: |                 if data["id"] in ["0", 0]: | ||||||
|                     # Skip invalid users |                     # Skip invalid users | ||||||
| @@ -568,14 +523,9 @@ class Client(object): | |||||||
|             "identifier": "thread_fbid", |             "identifier": "thread_fbid", | ||||||
|             "thread_fbid": thread_id, |             "thread_fbid": thread_id, | ||||||
|         } |         } | ||||||
|         j = self._post( |         j = self._payload_post("/ajax/mercury/search_snippets.php?dpr=1", data) | ||||||
|             "/ajax/mercury/search_snippets.php?dpr=1", |  | ||||||
|             data, |  | ||||||
|             fix_request=True, |  | ||||||
|             as_json=True, |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         result = j["payload"]["search_snippets"][query] |         result = j["search_snippets"][query] | ||||||
|         snippets = result[thread_id]["snippets"] if result.get(thread_id) else [] |         snippets = result[thread_id]["snippets"] if result.get(thread_id) else [] | ||||||
|         for snippet in snippets: |         for snippet in snippets: | ||||||
|             yield snippet["message_id"] |             yield snippet["message_id"] | ||||||
| @@ -618,13 +568,8 @@ class Client(object): | |||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         data = {"query": query, "snippetLimit": thread_limit} |         data = {"query": query, "snippetLimit": thread_limit} | ||||||
|         j = self._post( |         j = self._payload_post("/ajax/mercury/search_snippets.php?dpr=1", data) | ||||||
|             "/ajax/mercury/search_snippets.php?dpr=1", |         result = j["search_snippets"][query] | ||||||
|             data, |  | ||||||
|             fix_request=True, |  | ||||||
|             as_json=True, |  | ||||||
|         ) |  | ||||||
|         result = j["payload"]["search_snippets"][query] |  | ||||||
|  |  | ||||||
|         if not result: |         if not result: | ||||||
|             return {} |             return {} | ||||||
| @@ -641,14 +586,14 @@ class Client(object): | |||||||
|  |  | ||||||
|     def _fetchInfo(self, *ids): |     def _fetchInfo(self, *ids): | ||||||
|         data = {"ids[{}]".format(i): _id for i, _id in enumerate(ids)} |         data = {"ids[{}]".format(i): _id for i, _id in enumerate(ids)} | ||||||
|         j = self._post("/chat/user_info/", data, fix_request=True, as_json=True) |         j = self._payload_post("/chat/user_info/", data) | ||||||
|  |  | ||||||
|         if j.get("payload") is None or j["payload"].get("profiles") is None: |         if j.get("profiles") is None: | ||||||
|             raise FBchatException("No users/pages returned: {}".format(j)) |             raise FBchatException("No users/pages returned: {}".format(j)) | ||||||
|  |  | ||||||
|         entries = {} |         entries = {} | ||||||
|         for _id in j["payload"]["profiles"]: |         for _id in j["profiles"]: | ||||||
|             k = j["payload"]["profiles"][_id] |             k = j["profiles"][_id] | ||||||
|             if k["type"] in ["user", "friend"]: |             if k["type"] in ["user", "friend"]: | ||||||
|                 entries[_id] = { |                 entries[_id] = { | ||||||
|                     "id": _id, |                     "id": _id, | ||||||
| @@ -911,12 +856,10 @@ class Client(object): | |||||||
|             "last_action_timestamp": now() - 60 * 1000 |             "last_action_timestamp": now() - 60 * 1000 | ||||||
|             # 'last_action_timestamp': 0 |             # 'last_action_timestamp': 0 | ||||||
|         } |         } | ||||||
|         j = self._post( |         j = self._payload_post("/ajax/mercury/unread_threads.php", form) | ||||||
|             "/ajax/mercury/unread_threads.php", form, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         payload = j["payload"]["unread_thread_fbids"][0] |         result = j["unread_thread_fbids"][0] | ||||||
|         return payload["thread_fbids"] + payload["other_user_fbids"] |         return result["thread_fbids"] + result["other_user_fbids"] | ||||||
|  |  | ||||||
|     def fetchUnseen(self): |     def fetchUnseen(self): | ||||||
|         """ |         """ | ||||||
| @@ -926,12 +869,10 @@ class Client(object): | |||||||
|         :rtype: list |         :rtype: list | ||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         j = self._post( |         j = self._payload_post("/mercury/unseen_thread_ids/", None) | ||||||
|             "/mercury/unseen_thread_ids/", None, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         payload = j["payload"]["unseen_thread_fbids"][0] |         result = j["unseen_thread_fbids"][0] | ||||||
|         return payload["thread_fbids"] + payload["other_user_fbids"] |         return result["thread_fbids"] + result["other_user_fbids"] | ||||||
|  |  | ||||||
|     def fetchImageUrl(self, image_id): |     def fetchImageUrl(self, image_id): | ||||||
|         """Fetches the url to the original image from an image attachment ID |         """Fetches the url to the original image from an image attachment ID | ||||||
| @@ -944,9 +885,7 @@ class Client(object): | |||||||
|         """ |         """ | ||||||
|         image_id = str(image_id) |         image_id = str(image_id) | ||||||
|         data = {"photo_id": str(image_id)} |         data = {"photo_id": str(image_id)} | ||||||
|         j = self._get( |         j = self._post("/mercury/attachments/photo/", data) | ||||||
|             "/mercury/attachments/photo/", query=data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         url = get_jsmods_require(j, 3) |         url = get_jsmods_require(j, 3) | ||||||
|         if url is None: |         if url is None: | ||||||
| @@ -976,10 +915,8 @@ class Client(object): | |||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         data = {"question_id": poll_id} |         data = {"question_id": poll_id} | ||||||
|         j = self._post( |         j = self._payload_post("/ajax/mercury/get_poll_options", data) | ||||||
|             "/ajax/mercury/get_poll_options", data, fix_request=True, as_json=True |         return [PollOption._from_graphql(m) for m in j] | ||||||
|         ) |  | ||||||
|         return [PollOption._from_graphql(m) for m in j["payload"]] |  | ||||||
|  |  | ||||||
|     def fetchPlanInfo(self, plan_id): |     def fetchPlanInfo(self, plan_id): | ||||||
|         """ |         """ | ||||||
| @@ -991,8 +928,8 @@ class Client(object): | |||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         data = {"event_reminder_id": plan_id} |         data = {"event_reminder_id": plan_id} | ||||||
|         j = self._post("/ajax/eventreminder", data, fix_request=True, as_json=True) |         j = self._payload_post("/ajax/eventreminder", data) | ||||||
|         return Plan._from_fetch(j["payload"]) |         return Plan._from_fetch(j) | ||||||
|  |  | ||||||
|     def _getPrivateData(self): |     def _getPrivateData(self): | ||||||
|         j = self.graphql_request(GraphQL(doc_id="1868889766468115")) |         j = self.graphql_request(GraphQL(doc_id="1868889766468115")) | ||||||
| @@ -1116,7 +1053,7 @@ class Client(object): | |||||||
|  |  | ||||||
|     def _doSendRequest(self, data, get_thread_id=False): |     def _doSendRequest(self, data, get_thread_id=False): | ||||||
|         """Sends the data to `SendURL`, and returns the message ID or None on failure""" |         """Sends the data to `SendURL`, and returns the message ID or None on failure""" | ||||||
|         j = self._post("/messaging/send/", data, fix_request=True, as_json=True) |         j = self._post("/messaging/send/", data) | ||||||
|  |  | ||||||
|         # update JS token if received in response |         # update JS token if received in response | ||||||
|         fb_dtsg = get_jsmods_require(j, 2) |         fb_dtsg = get_jsmods_require(j, 2) | ||||||
| @@ -1251,8 +1188,7 @@ class Client(object): | |||||||
|         :param mid: :ref:`Message ID <intro_message_ids>` of the message to unsend |         :param mid: :ref:`Message ID <intro_message_ids>` of the message to unsend | ||||||
|         """ |         """ | ||||||
|         data = {"message_id": mid} |         data = {"message_id": mid} | ||||||
|         r = self._post("/messaging/unsend_message/?dpr=1", data) |         j = self._payload_post("/messaging/unsend_message/?dpr=1", data) | ||||||
|         r.raise_for_status() |  | ||||||
|  |  | ||||||
|     def _sendLocation( |     def _sendLocation( | ||||||
|         self, location, current=True, message=None, thread_id=None, thread_type=None |         self, location, current=True, message=None, thread_id=None, thread_type=None | ||||||
| @@ -1326,22 +1262,18 @@ class Client(object): | |||||||
|  |  | ||||||
|         data = {"voice_clip": voice_clip} |         data = {"voice_clip": voice_clip} | ||||||
|  |  | ||||||
|         j = self._postFile( |         j = self._payload_post( | ||||||
|             "https://upload.facebook.com/ajax/mercury/upload.php", |             "https://upload.facebook.com/ajax/mercury/upload.php", data, files=file_dict | ||||||
|             files=file_dict, |  | ||||||
|             query=data, |  | ||||||
|             fix_request=True, |  | ||||||
|             as_json=True, |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         if len(j["payload"]["metadata"]) != len(files): |         if len(j["metadata"]) != len(files): | ||||||
|             raise FBchatException( |             raise FBchatException( | ||||||
|                 "Some files could not be uploaded: {}, {}".format(j, files) |                 "Some files could not be uploaded: {}, {}".format(j, files) | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         return [ |         return [ | ||||||
|             (data[mimetype_to_key(data["filetype"])], data["filetype"]) |             (data[mimetype_to_key(data["filetype"])], data["filetype"]) | ||||||
|             for data in j["payload"]["metadata"] |             for data in j["metadata"] | ||||||
|         ] |         ] | ||||||
|  |  | ||||||
|     def _sendFiles( |     def _sendFiles( | ||||||
| @@ -1510,9 +1442,7 @@ class Client(object): | |||||||
|             "attachment_id": attachment_id, |             "attachment_id": attachment_id, | ||||||
|             "recipient_map[{}]".format(generateOfflineThreadingID()): thread_id, |             "recipient_map[{}]".format(generateOfflineThreadingID()): thread_id, | ||||||
|         } |         } | ||||||
|         j = self._post( |         j = self._payload_post("/mercury/attachments/forward/", data) | ||||||
|             "/mercury/attachments/forward/", data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def createGroup(self, message, user_ids): |     def createGroup(self, message, user_ids): | ||||||
|         """ |         """ | ||||||
| @@ -1578,9 +1508,7 @@ class Client(object): | |||||||
|         thread_id, thread_type = self._getThread(thread_id, None) |         thread_id, thread_type = self._getThread(thread_id, None) | ||||||
|  |  | ||||||
|         data = {"uid": user_id, "tid": thread_id} |         data = {"uid": user_id, "tid": thread_id} | ||||||
|         j = self._post( |         j = self._payload_post("/chat/remove_participants/", data) | ||||||
|             "/chat/remove_participants/", data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def _adminStatus(self, admin_ids, admin, thread_id=None): |     def _adminStatus(self, admin_ids, admin, thread_id=None): | ||||||
|         thread_id, thread_type = self._getThread(thread_id, None) |         thread_id, thread_type = self._getThread(thread_id, None) | ||||||
| @@ -1592,9 +1520,7 @@ class Client(object): | |||||||
|         for i, admin_id in enumerate(admin_ids): |         for i, admin_id in enumerate(admin_ids): | ||||||
|             data["admin_ids[{}]".format(i)] = str(admin_id) |             data["admin_ids[{}]".format(i)] = str(admin_id) | ||||||
|  |  | ||||||
|         j = self._post( |         j = self._payload_post("/messaging/save_admins/?dpr=1", data) | ||||||
|             "/messaging/save_admins/?dpr=1", data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def addGroupAdmins(self, admin_ids, thread_id=None): |     def addGroupAdmins(self, admin_ids, thread_id=None): | ||||||
|         """ |         """ | ||||||
| @@ -1627,9 +1553,7 @@ class Client(object): | |||||||
|         thread_id, thread_type = self._getThread(thread_id, None) |         thread_id, thread_type = self._getThread(thread_id, None) | ||||||
|  |  | ||||||
|         data = {"set_mode": int(require_admin_approval), "thread_fbid": thread_id} |         data = {"set_mode": int(require_admin_approval), "thread_fbid": thread_id} | ||||||
|         j = self._post( |         j = self._payload_post("/messaging/set_approval_mode/?dpr=1", data) | ||||||
|             "/messaging/set_approval_mode/?dpr=1", data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def _usersApproval(self, user_ids, approve, thread_id=None): |     def _usersApproval(self, user_ids, approve, thread_id=None): | ||||||
|         thread_id, thread_type = self._getThread(thread_id, None) |         thread_id, thread_type = self._getThread(thread_id, None) | ||||||
| @@ -1680,9 +1604,7 @@ class Client(object): | |||||||
|  |  | ||||||
|         data = {"thread_image_id": image_id, "thread_id": thread_id} |         data = {"thread_image_id": image_id, "thread_id": thread_id} | ||||||
|  |  | ||||||
|         j = self._post( |         j = self._payload_post("/messaging/set_thread_image/?dpr=1", data) | ||||||
|             "/messaging/set_thread_image/?dpr=1", data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|         return image_id |         return image_id | ||||||
|  |  | ||||||
|     def changeGroupImageRemote(self, image_url, thread_id=None): |     def changeGroupImageRemote(self, image_url, thread_id=None): | ||||||
| @@ -1729,9 +1651,7 @@ class Client(object): | |||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         data = {"thread_name": title, "thread_id": thread_id} |         data = {"thread_name": title, "thread_id": thread_id} | ||||||
|         j = self._post( |         j = self._payload_post("/messaging/set_thread_name/?dpr=1", data) | ||||||
|             "/messaging/set_thread_name/?dpr=1", data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def changeNickname( |     def changeNickname( | ||||||
|         self, nickname, user_id, thread_id=None, thread_type=ThreadType.USER |         self, nickname, user_id, thread_id=None, thread_type=ThreadType.USER | ||||||
| @@ -1753,11 +1673,8 @@ class Client(object): | |||||||
|             "participant_id": user_id, |             "participant_id": user_id, | ||||||
|             "thread_or_other_fbid": thread_id, |             "thread_or_other_fbid": thread_id, | ||||||
|         } |         } | ||||||
|         j = self._post( |         j = self._payload_post( | ||||||
|             "/messaging/save_thread_nickname/?source=thread_settings&dpr=1", |             "/messaging/save_thread_nickname/?source=thread_settings&dpr=1", data | ||||||
|             data, |  | ||||||
|             fix_request=True, |  | ||||||
|             as_json=True, |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def changeThreadColor(self, color, thread_id=None): |     def changeThreadColor(self, color, thread_id=None): | ||||||
| @@ -1775,11 +1692,8 @@ class Client(object): | |||||||
|             "color_choice": color.value if color != ThreadColor.MESSENGER_BLUE else "", |             "color_choice": color.value if color != ThreadColor.MESSENGER_BLUE else "", | ||||||
|             "thread_or_other_fbid": thread_id, |             "thread_or_other_fbid": thread_id, | ||||||
|         } |         } | ||||||
|         j = self._post( |         j = self._payload_post( | ||||||
|             "/messaging/save_thread_color/?source=thread_settings&dpr=1", |             "/messaging/save_thread_color/?source=thread_settings&dpr=1", data | ||||||
|             data, |  | ||||||
|             fix_request=True, |  | ||||||
|             as_json=True, |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def changeThreadEmoji(self, emoji, thread_id=None): |     def changeThreadEmoji(self, emoji, thread_id=None): | ||||||
| @@ -1795,11 +1709,8 @@ class Client(object): | |||||||
|         thread_id, thread_type = self._getThread(thread_id, None) |         thread_id, thread_type = self._getThread(thread_id, None) | ||||||
|  |  | ||||||
|         data = {"emoji_choice": emoji, "thread_or_other_fbid": thread_id} |         data = {"emoji_choice": emoji, "thread_or_other_fbid": thread_id} | ||||||
|         j = self._post( |         j = self._payload_post( | ||||||
|             "/messaging/save_thread_emoji/?source=thread_settings&dpr=1", |             "/messaging/save_thread_emoji/?source=thread_settings&dpr=1", data | ||||||
|             data, |  | ||||||
|             fix_request=True, |  | ||||||
|             as_json=True, |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def reactToMessage(self, message_id, reaction): |     def reactToMessage(self, message_id, reaction): | ||||||
| @@ -1819,7 +1730,7 @@ class Client(object): | |||||||
|             "reaction": reaction.value if reaction else None, |             "reaction": reaction.value if reaction else None, | ||||||
|         } |         } | ||||||
|         data = {"doc_id": 1491398900900362, "variables": json.dumps({"data": data})} |         data = {"doc_id": 1491398900900362, "variables": json.dumps({"data": data})} | ||||||
|         self._post("/webgraphql/mutation", data, fix_request=True, as_json=True) |         j = self._payload_post("/webgraphql/mutation", data) | ||||||
|  |  | ||||||
|     def createPlan(self, plan, thread_id=None): |     def createPlan(self, plan, thread_id=None): | ||||||
|         """ |         """ | ||||||
| @@ -1841,9 +1752,7 @@ class Client(object): | |||||||
|             "location_name": plan.location or "", |             "location_name": plan.location or "", | ||||||
|             "acontext": ACONTEXT, |             "acontext": ACONTEXT, | ||||||
|         } |         } | ||||||
|         j = self._post( |         j = self._payload_post("/ajax/eventreminder/create", data) | ||||||
|             "/ajax/eventreminder/create", data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def editPlan(self, plan, new_plan): |     def editPlan(self, plan, new_plan): | ||||||
|         """ |         """ | ||||||
| @@ -1863,9 +1772,7 @@ class Client(object): | |||||||
|             "title": new_plan.title, |             "title": new_plan.title, | ||||||
|             "acontext": ACONTEXT, |             "acontext": ACONTEXT, | ||||||
|         } |         } | ||||||
|         j = self._post( |         j = self._payload_post("/ajax/eventreminder/submit", data) | ||||||
|             "/ajax/eventreminder/submit", data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def deletePlan(self, plan): |     def deletePlan(self, plan): | ||||||
|         """ |         """ | ||||||
| @@ -1875,9 +1782,7 @@ class Client(object): | |||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         data = {"event_reminder_id": plan.uid, "delete": "true", "acontext": ACONTEXT} |         data = {"event_reminder_id": plan.uid, "delete": "true", "acontext": ACONTEXT} | ||||||
|         j = self._post( |         j = self._payload_post("/ajax/eventreminder/submit", data) | ||||||
|             "/ajax/eventreminder/submit", data, fix_request=True, as_json=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def changePlanParticipation(self, plan, take_part=True): |     def changePlanParticipation(self, plan, take_part=True): | ||||||
|         """ |         """ | ||||||
| @@ -1892,7 +1797,7 @@ class Client(object): | |||||||
|             "guest_state": "GOING" if take_part else "DECLINED", |             "guest_state": "GOING" if take_part else "DECLINED", | ||||||
|             "acontext": ACONTEXT, |             "acontext": ACONTEXT, | ||||||
|         } |         } | ||||||
|         j = self._post("/ajax/eventreminder/rsvp", data, fix_request=True, as_json=True) |         j = self._payload_post("/ajax/eventreminder/rsvp", data) | ||||||
|  |  | ||||||
|     def eventReminder(self, thread_id, time, title, location="", location_id=""): |     def eventReminder(self, thread_id, time, title, location="", location_id=""): | ||||||
|         """ |         """ | ||||||
| @@ -1922,12 +1827,7 @@ class Client(object): | |||||||
|             data["option_text_array[{}]".format(i)] = option.text |             data["option_text_array[{}]".format(i)] = option.text | ||||||
|             data["option_is_selected_array[{}]".format(i)] = str(int(option.vote)) |             data["option_is_selected_array[{}]".format(i)] = str(int(option.vote)) | ||||||
|  |  | ||||||
|         j = self._post( |         j = self._payload_post("/messaging/group_polling/create_poll/?dpr=1", data) | ||||||
|             "/messaging/group_polling/create_poll/?dpr=1", |  | ||||||
|             data, |  | ||||||
|             fix_request=True, |  | ||||||
|             as_json=True, |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def updatePollVote(self, poll_id, option_ids=[], new_options=[]): |     def updatePollVote(self, poll_id, option_ids=[], new_options=[]): | ||||||
|         """ |         """ | ||||||
| @@ -1949,12 +1849,7 @@ class Client(object): | |||||||
|         for i, option_text in enumerate(new_options): |         for i, option_text in enumerate(new_options): | ||||||
|             data["new_options[{}]".format(i)] = option_text |             data["new_options[{}]".format(i)] = option_text | ||||||
|  |  | ||||||
|         j = self._post( |         j = self._payload_post("/messaging/group_polling/update_vote/?dpr=1", data) | ||||||
|             "/messaging/group_polling/update_vote/?dpr=1", |  | ||||||
|             data, |  | ||||||
|             fix_request=True, |  | ||||||
|             as_json=True, |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def setTypingStatus(self, status, thread_id=None, thread_type=None): |     def setTypingStatus(self, status, thread_id=None, thread_type=None): | ||||||
|         """ |         """ | ||||||
| @@ -1975,7 +1870,7 @@ class Client(object): | |||||||
|             "to": thread_id if thread_type == ThreadType.USER else "", |             "to": thread_id if thread_type == ThreadType.USER else "", | ||||||
|             "source": "mercury-chat", |             "source": "mercury-chat", | ||||||
|         } |         } | ||||||
|         j = self._post("/ajax/messaging/typ.php", data, fix_request=True, as_json=True) |         j = self._payload_post("/ajax/messaging/typ.php", data) | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     END SEND METHODS |     END SEND METHODS | ||||||
| @@ -1987,7 +1882,7 @@ class Client(object): | |||||||
|  |  | ||||||
|         :param thread_id: User/Group ID to which the message belongs. See :ref:`intro_threads` |         :param thread_id: User/Group ID to which the message belongs. See :ref:`intro_threads` | ||||||
|         :param message_id: Message ID to set as delivered. See :ref:`intro_threads` |         :param message_id: Message ID to set as delivered. See :ref:`intro_threads` | ||||||
|         :return: Whether the request was successful |         :return: True | ||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         data = { |         data = { | ||||||
| @@ -1995,8 +1890,8 @@ class Client(object): | |||||||
|             "thread_ids[%s][0]" % thread_id: message_id, |             "thread_ids[%s][0]" % thread_id: message_id, | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         r = self._post("/ajax/mercury/delivery_receipts.php", data) |         j = self._payload_post("/ajax/mercury/delivery_receipts.php", data) | ||||||
|         return r.ok |         return True | ||||||
|  |  | ||||||
|     def _readStatus(self, read, thread_ids): |     def _readStatus(self, read, thread_ids): | ||||||
|         thread_ids = require_list(thread_ids) |         thread_ids = require_list(thread_ids) | ||||||
| @@ -2006,8 +1901,7 @@ class Client(object): | |||||||
|         for thread_id in thread_ids: |         for thread_id in thread_ids: | ||||||
|             data["ids[{}]".format(thread_id)] = "true" if read else "false" |             data["ids[{}]".format(thread_id)] = "true" if read else "false" | ||||||
|  |  | ||||||
|         r = self._post("/ajax/mercury/change_read_status.php", data) |         j = self._payload_post("/ajax/mercury/change_read_status.php", data) | ||||||
|         return r.ok |  | ||||||
|  |  | ||||||
|     def markAsRead(self, thread_ids=None): |     def markAsRead(self, thread_ids=None): | ||||||
|         """ |         """ | ||||||
| @@ -2015,7 +1909,6 @@ class Client(object): | |||||||
|         All messages inside the threads will be marked as read |         All messages inside the threads will be marked as read | ||||||
|  |  | ||||||
|         :param thread_ids: User/Group IDs to set as read. See :ref:`intro_threads` |         :param thread_ids: User/Group IDs to set as read. See :ref:`intro_threads` | ||||||
|         :return: Whether the request was successful |  | ||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         self._readStatus(True, thread_ids) |         self._readStatus(True, thread_ids) | ||||||
| @@ -2026,7 +1919,6 @@ class Client(object): | |||||||
|         All messages inside the threads will be marked as unread |         All messages inside the threads will be marked as unread | ||||||
|  |  | ||||||
|         :param thread_ids: User/Group IDs to set as unread. See :ref:`intro_threads` |         :param thread_ids: User/Group IDs to set as unread. See :ref:`intro_threads` | ||||||
|         :return: Whether the request was successful |  | ||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         self._readStatus(False, thread_ids) |         self._readStatus(False, thread_ids) | ||||||
| @@ -2036,8 +1928,7 @@ class Client(object): | |||||||
|         .. todo:: |         .. todo:: | ||||||
|             Documenting this |             Documenting this | ||||||
|         """ |         """ | ||||||
|         r = self._post("/ajax/mercury/mark_seen.php", {"seen_timestamp": now()}) |         j = self._payload_post("/ajax/mercury/mark_seen.php", {"seen_timestamp": now()}) | ||||||
|         return r.ok |  | ||||||
|  |  | ||||||
|     def friendConnect(self, friend_id): |     def friendConnect(self, friend_id): | ||||||
|         """ |         """ | ||||||
| @@ -2046,37 +1937,31 @@ class Client(object): | |||||||
|         """ |         """ | ||||||
|         data = {"to_friend": friend_id, "action": "confirm"} |         data = {"to_friend": friend_id, "action": "confirm"} | ||||||
|  |  | ||||||
|         r = self._post("/ajax/add_friend/action.php?dpr=1", data) |         j = self._payload_post("/ajax/add_friend/action.php?dpr=1", data) | ||||||
|         return r.ok |  | ||||||
|  |  | ||||||
|     def removeFriend(self, friend_id=None): |     def removeFriend(self, friend_id=None): | ||||||
|         """ |         """ | ||||||
|         Removes a specifed friend from your friend list |         Removes a specifed friend from your friend list | ||||||
|  |  | ||||||
|         :param friend_id: The ID of the friend that you want to remove |         :param friend_id: The ID of the friend that you want to remove | ||||||
|         :return: Returns error if the removing was unsuccessful, returns True when successful. |         :return: True | ||||||
|  |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         payload = {"friend_id": friend_id, "unref": "none", "confirm": "Confirm"} |         data = {"uid": friend_id} | ||||||
|         r = self._post("https://m.facebook.com/a/removefriend.php", payload) |         j = self._payload_post("/ajax/profile/removefriendconfirm.php", data) | ||||||
|         query = parse_qs(urlparse(r.url).query) |  | ||||||
|         if "err" not in query: |  | ||||||
|             log.debug("Remove was successful!") |  | ||||||
|         return True |         return True | ||||||
|         else: |  | ||||||
|             log.warning("Error while removing friend") |  | ||||||
|             return False |  | ||||||
|  |  | ||||||
|     def blockUser(self, user_id): |     def blockUser(self, user_id): | ||||||
|         """ |         """ | ||||||
|         Blocks messages from a specifed user |         Blocks messages from a specifed user | ||||||
|  |  | ||||||
|         :param user_id: The ID of the user that you want to block |         :param user_id: The ID of the user that you want to block | ||||||
|         :return: Whether the request was successful |         :return: True | ||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         data = {"fbid": user_id} |         data = {"fbid": user_id} | ||||||
|         r = self._post("/messaging/block_messages/?dpr=1", data) |         j = self._payload_post("/messaging/block_messages/?dpr=1", data) | ||||||
|         return r.ok |         return True | ||||||
|  |  | ||||||
|     def unblockUser(self, user_id): |     def unblockUser(self, user_id): | ||||||
|         """ |         """ | ||||||
| @@ -2087,8 +1972,8 @@ class Client(object): | |||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         data = {"fbid": user_id} |         data = {"fbid": user_id} | ||||||
|         r = self._post("/messaging/unblock_messages/?dpr=1", data) |         j = self._payload_post("/messaging/unblock_messages/?dpr=1", data) | ||||||
|         return r.ok |         return True | ||||||
|  |  | ||||||
|     def moveThreads(self, location, thread_ids): |     def moveThreads(self, location, thread_ids): | ||||||
|         """ |         """ | ||||||
| @@ -2096,7 +1981,7 @@ class Client(object): | |||||||
|  |  | ||||||
|         :param location: models.ThreadLocation: INBOX, PENDING, ARCHIVED or OTHER |         :param location: models.ThreadLocation: INBOX, PENDING, ARCHIVED or OTHER | ||||||
|         :param thread_ids: Thread IDs to move. See :ref:`intro_threads` |         :param thread_ids: Thread IDs to move. See :ref:`intro_threads` | ||||||
|         :return: Whether the request was successful |         :return: True | ||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         thread_ids = require_list(thread_ids) |         thread_ids = require_list(thread_ids) | ||||||
| @@ -2110,26 +1995,25 @@ class Client(object): | |||||||
|             for thread_id in thread_ids: |             for thread_id in thread_ids: | ||||||
|                 data_archive["ids[{}]".format(thread_id)] = "true" |                 data_archive["ids[{}]".format(thread_id)] = "true" | ||||||
|                 data_unpin["ids[{}]".format(thread_id)] = "false" |                 data_unpin["ids[{}]".format(thread_id)] = "false" | ||||||
|             r_archive = self._post( |             j_archive = self._payload_post( | ||||||
|                 "/ajax/mercury/change_archived_status.php?dpr=1", data_archive |                 "/ajax/mercury/change_archived_status.php?dpr=1", data_archive | ||||||
|             ) |             ) | ||||||
|             r_unpin = self._post( |             j_unpin = self._payload_post( | ||||||
|                 "/ajax/mercury/change_pinned_status.php?dpr=1", data_unpin |                 "/ajax/mercury/change_pinned_status.php?dpr=1", data_unpin | ||||||
|             ) |             ) | ||||||
|             return r_archive.ok and r_unpin.ok |  | ||||||
|         else: |         else: | ||||||
|             data = dict() |             data = dict() | ||||||
|             for i, thread_id in enumerate(thread_ids): |             for i, thread_id in enumerate(thread_ids): | ||||||
|                 data["{}[{}]".format(location.name.lower(), i)] = thread_id |                 data["{}[{}]".format(location.name.lower(), i)] = thread_id | ||||||
|             r = self._post("/ajax/mercury/move_thread.php", data) |             j = self._payload_post("/ajax/mercury/move_thread.php", data) | ||||||
|             return r.ok |         return True | ||||||
|  |  | ||||||
|     def deleteThreads(self, thread_ids): |     def deleteThreads(self, thread_ids): | ||||||
|         """ |         """ | ||||||
|         Deletes threads |         Deletes threads | ||||||
|  |  | ||||||
|         :param thread_ids: Thread IDs to delete. See :ref:`intro_threads` |         :param thread_ids: Thread IDs to delete. See :ref:`intro_threads` | ||||||
|         :return: Whether the request was successful |         :return: True | ||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         thread_ids = require_list(thread_ids) |         thread_ids = require_list(thread_ids) | ||||||
| @@ -2139,36 +2023,40 @@ class Client(object): | |||||||
|         for i, thread_id in enumerate(thread_ids): |         for i, thread_id in enumerate(thread_ids): | ||||||
|             data_unpin["ids[{}]".format(thread_id)] = "false" |             data_unpin["ids[{}]".format(thread_id)] = "false" | ||||||
|             data_delete["ids[{}]".format(i)] = thread_id |             data_delete["ids[{}]".format(i)] = thread_id | ||||||
|         r_unpin = self._post("/ajax/mercury/change_pinned_status.php?dpr=1", data_unpin) |         j_unpin = self._payload_post( | ||||||
|         r_delete = self._post("/ajax/mercury/delete_thread.php?dpr=1", data_delete) |             "/ajax/mercury/change_pinned_status.php?dpr=1", data_unpin | ||||||
|         return r_unpin.ok and r_delete.ok |         ) | ||||||
|  |         j_delete = self._payload_post( | ||||||
|  |             "/ajax/mercury/delete_thread.php?dpr=1", data_delete | ||||||
|  |         ) | ||||||
|  |         return True | ||||||
|  |  | ||||||
|     def markAsSpam(self, thread_id=None): |     def markAsSpam(self, thread_id=None): | ||||||
|         """ |         """ | ||||||
|         Mark a thread as spam and delete it |         Mark a thread as spam and delete it | ||||||
|  |  | ||||||
|         :param thread_id: User/Group ID to mark as spam. See :ref:`intro_threads` |         :param thread_id: User/Group ID to mark as spam. See :ref:`intro_threads` | ||||||
|         :return: Whether the request was successful |         :return: True | ||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         thread_id, thread_type = self._getThread(thread_id, None) |         thread_id, thread_type = self._getThread(thread_id, None) | ||||||
|         r = self._post("/ajax/mercury/mark_spam.php?dpr=1", {"id": thread_id}) |         j = self._payload_post("/ajax/mercury/mark_spam.php?dpr=1", {"id": thread_id}) | ||||||
|         return r.ok |         return True | ||||||
|  |  | ||||||
|     def deleteMessages(self, message_ids): |     def deleteMessages(self, message_ids): | ||||||
|         """ |         """ | ||||||
|         Deletes specifed messages |         Deletes specifed messages | ||||||
|  |  | ||||||
|         :param message_ids: Message IDs to delete |         :param message_ids: Message IDs to delete | ||||||
|         :return: Whether the request was successful |         :return: True | ||||||
|         :raises: FBchatException if request failed |         :raises: FBchatException if request failed | ||||||
|         """ |         """ | ||||||
|         message_ids = require_list(message_ids) |         message_ids = require_list(message_ids) | ||||||
|         data = dict() |         data = dict() | ||||||
|         for i, message_id in enumerate(message_ids): |         for i, message_id in enumerate(message_ids): | ||||||
|             data["message_ids[{}]".format(i)] = message_id |             data["message_ids[{}]".format(i)] = message_id | ||||||
|         r = self._post("/ajax/mercury/delete_messages.php?dpr=1", data) |         j = self._payload_post("/ajax/mercury/delete_messages.php?dpr=1", data) | ||||||
|         return r.ok |         return True | ||||||
|  |  | ||||||
|     def muteThread(self, mute_time=-1, thread_id=None): |     def muteThread(self, mute_time=-1, thread_id=None): | ||||||
|         """ |         """ | ||||||
| @@ -2179,9 +2067,7 @@ class Client(object): | |||||||
|         """ |         """ | ||||||
|         thread_id, thread_type = self._getThread(thread_id, None) |         thread_id, thread_type = self._getThread(thread_id, None) | ||||||
|         data = {"mute_settings": str(mute_time), "thread_fbid": thread_id} |         data = {"mute_settings": str(mute_time), "thread_fbid": thread_id} | ||||||
|         content = self._post( |         j = self._payload_post("/ajax/mercury/change_mute_thread.php?dpr=1", data) | ||||||
|             "/ajax/mercury/change_mute_thread.php?dpr=1", data, fix_request=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def unmuteThread(self, thread_id=None): |     def unmuteThread(self, thread_id=None): | ||||||
|         """ |         """ | ||||||
| @@ -2200,8 +2086,8 @@ class Client(object): | |||||||
|         """ |         """ | ||||||
|         thread_id, thread_type = self._getThread(thread_id, None) |         thread_id, thread_type = self._getThread(thread_id, None) | ||||||
|         data = {"reactions_mute_mode": int(mute), "thread_fbid": thread_id} |         data = {"reactions_mute_mode": int(mute), "thread_fbid": thread_id} | ||||||
|         r = self._post( |         j = self._payload_post( | ||||||
|             "/ajax/mercury/change_reactions_mute_thread/?dpr=1", data, fix_request=True |             "/ajax/mercury/change_reactions_mute_thread/?dpr=1", data | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def unmuteThreadReactions(self, thread_id=None): |     def unmuteThreadReactions(self, thread_id=None): | ||||||
| @@ -2221,9 +2107,7 @@ class Client(object): | |||||||
|         """ |         """ | ||||||
|         thread_id, thread_type = self._getThread(thread_id, None) |         thread_id, thread_type = self._getThread(thread_id, None) | ||||||
|         data = {"mentions_mute_mode": int(mute), "thread_fbid": thread_id} |         data = {"mentions_mute_mode": int(mute), "thread_fbid": thread_id} | ||||||
|         r = self._post( |         j = self._payload_post("/ajax/mercury/change_mentions_mute_thread/?dpr=1", data) | ||||||
|             "/ajax/mercury/change_mentions_mute_thread/?dpr=1", data, fix_request=True |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def unmuteThreadMentions(self, thread_id=None): |     def unmuteThreadMentions(self, thread_id=None): | ||||||
|         """ |         """ | ||||||
| @@ -2250,11 +2134,9 @@ class Client(object): | |||||||
|             "viewer_uid": self._uid, |             "viewer_uid": self._uid, | ||||||
|             "state": "active", |             "state": "active", | ||||||
|         } |         } | ||||||
|         self._get( |         j = self._get( | ||||||
|             "https://{}-edge-chat.facebook.com/active_ping".format(self._pull_channel), |             "https://{}-edge-chat.facebook.com/active_ping".format(self._pull_channel), | ||||||
|             data, |             data, | ||||||
|             fix_request=True, |  | ||||||
|             as_json=False, |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def _pullMessage(self): |     def _pullMessage(self): | ||||||
| @@ -2268,10 +2150,7 @@ class Client(object): | |||||||
|             "state": "active" if self._markAlive else "offline", |             "state": "active" if self._markAlive else "offline", | ||||||
|         } |         } | ||||||
|         return self._get( |         return self._get( | ||||||
|             "https://{}-edge-chat.facebook.com/pull".format(self._pull_channel), |             "https://{}-edge-chat.facebook.com/pull".format(self._pull_channel), data | ||||||
|             data, |  | ||||||
|             fix_request=True, |  | ||||||
|             as_json=True, |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def _parseDelta(self, m): |     def _parseDelta(self, m): | ||||||
|   | |||||||
| @@ -140,25 +140,31 @@ def check_json(j): | |||||||
|  |  | ||||||
|  |  | ||||||
| def check_request(r, as_json=True): | def check_request(r, as_json=True): | ||||||
|     if not r.ok: |     check_http_code(r.status_code) | ||||||
|  |     content = get_decoded_r(r) | ||||||
|  |     check_content(content) | ||||||
|  |     return to_json(content) if as_json else content | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def check_http_code(code): | ||||||
|  |     if 400 <= code < 600: | ||||||
|         raise FBchatFacebookError( |         raise FBchatFacebookError( | ||||||
|             "Error when sending request: Got {} response".format(r.status_code), |             "Error when sending request: Got {} response".format(code), | ||||||
|             request_status_code=r.status_code, |             request_status_code=code, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     content = get_decoded_r(r) |  | ||||||
|  |  | ||||||
|  | def check_content(content, as_json=True): | ||||||
|     if content is None or len(content) == 0: |     if content is None or len(content) == 0: | ||||||
|         raise FBchatFacebookError("Error when sending request: Got empty response") |         raise FBchatFacebookError("Error when sending request: Got empty response") | ||||||
|  |  | ||||||
|     if as_json: |  | ||||||
|  | def to_json(content): | ||||||
|     content = strip_json_cruft(content) |     content = strip_json_cruft(content) | ||||||
|     j = parse_json(content) |     j = parse_json(content) | ||||||
|     check_json(j) |     check_json(j) | ||||||
|     log.debug(j) |     log.debug(j) | ||||||
|     return j |     return j | ||||||
|     else: |  | ||||||
|         return content |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_jsmods_require(j, index): | def get_jsmods_require(j, index): | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user