def test_title_set(session):
    data = {
        "irisSeqId": "11223344",
        "irisTags": ["DeltaThreadName", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "3456",
            "adminText": "You named the group abc.",
            "folderId": {"systemFolderId": "INBOX"},
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "1122334455",
            "skipBumpThread": False,
            "tags": [],
            "threadKey": {"threadFbId": "4321"},
            "threadReadStateEffect": "KEEP_AS_IS",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "name": "abc",
        "participants": ["1234", "2345", "3456", "4567"],
        "requestContext": {"apiArgs": {}},
        "tqSeqId": "1111",
        "class": "ThreadName",
    }
    assert TitleSet(
        author=User(session=session, id="3456"),
        thread=Group(session=session, id="4321"),
        title="abc",
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_delta(session, data)
def test_plan_ended(session):
    data = {}
    assert PlanEnded(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        plan=PlanData(
            session=session,
            id="112233",
            time=datetime.datetime(2020,
                                   9,
                                   13,
                                   12,
                                   26,
                                   40,
                                   tzinfo=datetime.timezone.utc),
            title="A plan",
            author_id="1234",
            guests={
                "1234": GuestStatus.GOING,
                "2345": GuestStatus.INVITED,
                "3456": GuestStatus.INVITED,
            },
        ),
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_admin_message(session, data)
def test_person_removed(session):
    data = {
        "irisSeqId": "11223344",
        "irisTags": ["DeltaParticipantLeftGroupThread", "is_from_iris_fanout"],
        "leftParticipantFbId": "1234",
        "messageMetadata": {
            "actorFbId": "3456",
            "adminText": "You removed Abc Def from the group.",
            "folderId": {"systemFolderId": "INBOX"},
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "1122334455",
            "skipBumpThread": True,
            "tags": [],
            "threadKey": {"threadFbId": "4321"},
            "threadReadStateEffect": "KEEP_AS_IS",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456", "4567"],
        "requestContext": {"apiArgs": {}},
        "tqSeqId": "1111",
        "class": "ParticipantLeftGroupThread",
    }
    assert PersonRemoved(
        author=User(session=session, id="3456"),
        thread=Group(session=session, id="4321"),
        removed=User(session=session, id="1234"),
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_delta(session, data)
Beispiel #4
0
def test_thread_typing(session):
    data = {"sender_fbid": 1234, "state": 0, "type": "typ", "thread": "4321"}
    (event, ) = parse_events(session, "/thread_typing", data)
    assert event == Typing(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        status=False,
    )
Beispiel #5
0
def test_t_ms_full(session):
    """A full example of parsing of data in /t_ms."""
    payload = {
        "deltas": [{
            "deltaMessageReaction": {
                "threadKey": {
                    "threadFbId": 4321
                },
                "messageId": "mid.$XYZ",
                "action": 0,
                "userId": 1234,
                "reaction": "😢",
                "senderId": 1234,
                "offlineThreadingId": "1122334455",
            }
        }]
    }
    data = {
        "deltas": [
            {
                "payload": [ord(x) for x in _util.json_minimal(payload)],
                "class": "ClientPayload",
            },
            {
                "class": "NoOp",
            },
            {
                "forceInsert": False,
                "messageId": "mid.$ABC",
                "threadKey": {
                    "threadFbId": "4321"
                },
                "class": "ForcedFetch",
            },
        ],
        "firstDeltaSeqId":
        111111,
        "lastIssuedSeqId":
        111113,
        "queueEntityId":
        1234,
    }
    thread = Group(session=session, id="4321")
    assert [
        ReactionEvent(
            author=User(session=session, id="1234"),
            thread=thread,
            message=Message(thread=thread, id="mid.$XYZ"),
            reaction="😢",
        ),
        UnfetchedThreadEvent(
            thread=thread,
            message=Message(thread=thread, id="mid.$ABC"),
        ),
    ] == list(parse_events(session, "/t_ms", data))
def test_forced_fetch(session):
    data = {
        "forceInsert": False,
        "messageId": "mid.$XYZ",
        "threadKey": {"threadFbId": "1234"},
        "class": "ForcedFetch",
    }
    thread = Group(session=session, id="1234")
    assert UnfetchedThreadEvent(
        thread=thread, message=Message(thread=thread, id="mid.$XYZ")
    ) == parse_delta(session, data)
def test_forced_fetch_pending(session):
    data = {
        "forceInsert": False,
        "irisSeqId": "1111",
        "isLazy": False,
        "threadKey": {"threadFbId": "1234"},
        "class": "ForcedFetch",
    }
    assert UnfetchedThreadEvent(
        thread=Group(session=session, id="1234"), message=None
    ) == parse_delta(session, data)
Beispiel #8
0
def test_event_get_thread_group2(session):
    data = {
        "actorFbId": "4321",
        "folderId": {"systemFolderId": "INBOX"},
        "messageId": "mid.$XYZ",
        "offlineThreadingId": "112233445566",
        "skipBumpThread": False,
        "tags": ["source:messenger:web"],
        "threadKey": {"threadFbId": "1234"},
        "threadReadStateEffect": "KEEP_AS_IS",
        "timestamp": "1500000000000",
    }
    assert Group(session=session, id="1234") == Event._get_thread(session, data)
Beispiel #9
0
def test_thread_parse_participants(session):
    nodes = [
        {"messaging_actor": {"__typename": "User", "id": "1234"}},
        {"messaging_actor": {"__typename": "User", "id": "2345"}},
        {"messaging_actor": {"__typename": "Page", "id": "3456"}},
        {"messaging_actor": {"__typename": "MessageThread", "id": "4567"}},
        {"messaging_actor": {"__typename": "UnavailableMessagingActor", "id": "5678"}},
    ]
    assert [
        User(session=session, id="1234"),
        User(session=session, id="2345"),
        Page(session=session, id="3456"),
        Group(session=session, id="4567"),
    ] == list(ThreadABC._parse_participants(session, {"nodes": nodes}))
def test_mark_read(session):
    data = {
        "actionTimestamp": "1600000000000",
        "irisSeqId": "1111111",
        "irisTags": ["DeltaMarkRead", "is_from_iris_fanout"],
        "threadKeys": [{"threadFbId": "1234"}, {"otherUserFbId": "2345"}],
        "tqSeqId": "1111",
        "watermarkTimestamp": "1500000000000",
        "class": "MarkRead",
    }
    assert ThreadsRead(
        author=User(session=session, id=session.user_id),
        threads=[Group(session=session, id="1234"), User(session=session, id="2345")],
        at=datetime.datetime(2020, 9, 13, 12, 26, 40, tzinfo=datetime.timezone.utc),
    ) == parse_delta(session, data)
def test_new_message_group(session):
    data = {
        "attachments": [],
        "body": "test",
        "irisSeqId": "1111111",
        "irisTags": ["DeltaNewMessage", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "4321",
            "folderId": {"systemFolderId": "INBOX"},
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": False,
            "tags": ["source:messenger:web"],
            "threadKey": {"threadFbId": "1234"},
            "threadReadStateEffect": "KEEP_AS_IS",
            "timestamp": "1600000000000",
        },
        "participants": ["4321", "5432", "6543"],
        "requestContext": {"apiArgs": {}},
        "tqSeqId": "1111",
        "class": "NewMessage",
    }
    assert MessageEvent(
        author=User(session=session, id="4321"),
        thread=Group(session=session, id="1234"),
        message=MessageData(
            thread=Group(session=session, id="1234"),
            id="mid.$XYZ",
            author="4321",
            text="test",
            created_at=datetime.datetime(
                2020, 9, 13, 12, 26, 40, tzinfo=datetime.timezone.utc
            ),
        ),
        at=datetime.datetime(2020, 9, 13, 12, 26, 40, tzinfo=datetime.timezone.utc),
    ) == parse_delta(session, data)
Beispiel #12
0
def test_event_get_thread_group1(session):
    data = {
        "threadKey": {"threadFbId": 1234},
        "messageId": "mid.$gAAT4Sw1WSGh14A3MOFvrsiDvr3Yc",
        "offlineThreadingId": "6623583531508397596",
        "actorFbId": 4321,
        "timestamp": 1500000000000,
        "tags": [
            "inbox",
            "sent",
            "tq",
            "blindly_apply_message_folder",
            "source:messenger:web",
        ],
    }
    assert Group(session=session, id="1234") == Event._get_thread(session, data)
def test_read_receipt(session):
    data = {
        "actionTimestampMs": "1600000000000",
        "actorFbId": "1234",
        "irisSeqId": "1111111",
        "irisTags": ["DeltaReadReceipt", "is_from_iris_fanout"],
        "requestContext": {"apiArgs": {}},
        "threadKey": {"threadFbId": "4321"},
        "tqSeqId": "1111",
        "watermarkTimestampMs": "1500000000000",
        "class": "ReadReceipt",
    }
    assert ThreadsRead(
        author=User(session=session, id="1234"),
        threads=[Group(session=session, id="4321")],
        at=datetime.datetime(2020, 9, 13, 12, 26, 40, tzinfo=datetime.timezone.utc),
    ) == parse_delta(session, data)
Beispiel #14
0
async def handle_got_message(msg, session, fbchat_client):
    logging.info(f"API Message: {msg}")
    if type(msg) == str:
        resp_json = loads(msg)
    else:
        resp_json = msg
    got_gateway = resp_json.get("gateway")
    if bool(got_gateway) is True:
        got_username = resp_json.get("username")
        file_data = None
        try:
            file_data = resp_json["Extra"]["file"][0]["Data"]
            file_data = standard_b64decode(file_data)
            search_link = False
            got_text = resp_json["Extra"]["file"][0]["Name"]
        except (KeyError, TypeError):
            logging.info(f"From API received json: {resp_json}")
            search_link = True
            got_text = resp_json.get("text")

        img_type_result, filename, cat = await find_file_type(
            search_text=got_text, search_link=search_link)

        if filename == got_text and search_link is False:
            got_text = f"sent {img_type_result} file"

        fb_thread = NeededVars.reverse_threads[got_gateway]

        thread = Group(session=session, id=fb_thread)
        if fb_thread in NeededVars.users:
            thread = User(session=session, id=fb_thread)

        await send_file(f"{got_username}", thread, fbchat_client, file_data,
                        cat, img_type_result, filename, search_link)

        got_text = await format_text_quote(got_text)

        logging.info(
            f"From api sending message: username: {got_username} | text: {got_text}"
        )

        await send_text(f"{got_username}{got_text}", thread)

        logging.info(
            f"Sent message: username: {got_username} | text: {got_text}")
Beispiel #15
0
def test_reaction_event_removed(session):
    data = {
        "threadKey": {
            "threadFbId": 1234
        },
        "messageId": "mid.$XYZ",
        "action": 1,
        "userId": 4321,
        "senderId": 4321,
        "offlineThreadingId": "6623586106713014836",
    }
    thread = Group(session=session, id="1234")
    assert ReactionEvent(
        author=User(session=session, id="4321"),
        thread=thread,
        message=Message(thread=thread, id="mid.$XYZ"),
        reaction=None,
    ) == parse_client_delta(session, {"deltaMessageReaction": data})
def test_admins_removed(session):
    data = {
        "irisSeqId": "1111111",
        "irisTags": ["DeltaAdminTextMessage", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "1234",
            "adminText": "You removed yourself as a group admin.",
            "folderId": {
                "systemFolderId": "INBOX"
            },
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": True,
            "tags": ["source:titan:web"],
            "threadKey": {
                "threadFbId": "4321"
            },
            "threadReadStateEffect": "KEEP_AS_IS",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456"],
        "requestContext": {
            "apiArgs": {}
        },
        "tqSeqId": "1111",
        "type": "change_thread_admins",
        "untypedData": {
            "THREAD_CATEGORY": "GROUP",
            "TARGET_ID": "1234",
            "ADMIN_TYPE": "0",
            "ADMIN_EVENT": "remove_admin",
        },
        "class": "AdminTextMessage",
    }
    assert AdminsRemoved(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        removed=[User(session=session, id="1234")],
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_admin_message(session, data)
def test_delivery_receipt_group(session):
    data = {
        "actorFbId": "1234",
        "deliveredWatermarkTimestampMs": "1500000000000",
        "irisSeqId": "1111111",
        "irisTags": ["DeltaDeliveryReceipt"],
        "messageIds": ["mid.$XYZ", "mid.$ABC"],
        "requestContext": {"apiArgs": {}},
        "threadKey": {"threadFbId": "4321"},
        "class": "DeliveryReceipt",
    }
    thread = Group(session=session, id="4321")
    assert MessagesDelivered(
        author=User(session=session, id="1234"),
        thread=thread,
        messages=[
            Message(thread=thread, id="mid.$XYZ"),
            Message(thread=thread, id="mid.$ABC"),
        ],
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_delta(session, data)
def test_color_set(session):
    data = {
        "irisSeqId": "1111111",
        "irisTags": ["DeltaAdminTextMessage", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "1234",
            "adminText": "You changed the chat theme to Orange.",
            "folderId": {
                "systemFolderId": "INBOX"
            },
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": False,
            "tags": ["source:titan:web", "no_push"],
            "threadKey": {
                "threadFbId": "4321"
            },
            "threadReadStateEffect": "MARK_UNREAD",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456"],
        "requestContext": {
            "apiArgs": {}
        },
        "tqSeqId": "1111",
        "type": "change_thread_theme",
        "untypedData": {
            "should_show_icon": "1",
            "theme_color": "FFFF7E29",
            "accessibility_label": "Orange",
        },
        "class": "AdminTextMessage",
    }
    assert ColorSet(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        color="#ff7e29",
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_admin_message(session, data)
def test_approvalmode_unset(session):
    data = {
        "irisSeqId": "1111111",
        "irisTags": ["DeltaAdminTextMessage", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "1234",
            "adminText":
            "You turned off member approval. Anyone with the link can join the group.",
            "folderId": {
                "systemFolderId": "INBOX"
            },
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": True,
            "tags": ["source:titan:web", "no_push"],
            "threadKey": {
                "threadFbId": "4321"
            },
            "threadReadStateEffect": "KEEP_AS_IS",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456"],
        "requestContext": {
            "apiArgs": {}
        },
        "tqSeqId": "1111",
        "type": "change_thread_approval_mode",
        "untypedData": {
            "APPROVAL_MODE": "0",
            "THREAD_CATEGORY": "GROUP"
        },
        "class": "AdminTextMessage",
    }
    assert ApprovalModeSet(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        require_admin_approval=False,
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_admin_message(session, data)
def test_nickname_set(session):
    data = {
        "irisSeqId": "1111111",
        "irisTags": ["DeltaAdminTextMessage", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "1234",
            "adminText": "You set the nickname for Abc Def to abc.",
            "folderId": {
                "systemFolderId": "INBOX"
            },
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": False,
            "tags": ["source:titan:web", "no_push"],
            "threadKey": {
                "threadFbId": "4321"
            },
            "threadReadStateEffect": "MARK_UNREAD",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456"],
        "requestContext": {
            "apiArgs": {}
        },
        "tqSeqId": "1111",
        "type": "change_thread_nickname",
        "untypedData": {
            "nickname": "abc",
            "participant_id": "2345"
        },
        "class": "AdminTextMessage",
    }
    assert NicknameSet(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        subject=User(session=session, id="2345"),
        nickname="abc",
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_admin_message(session, data)
def test_call_joined(session):
    data = {
        "irisSeqId": "1111111",
        "irisTags": ["DeltaAdminTextMessage", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "1234",
            "adminText": "Abc joined the call.",
            "folderId": {
                "systemFolderId": "INBOX"
            },
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": False,
            "tags": ["source:titan:web"],
            "threadKey": {
                "threadFbId": "4321"
            },
            "threadReadStateEffect": "MARK_UNREAD",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456"],
        "requestContext": {
            "apiArgs": {}
        },
        "tqSeqId": "1111",
        "type": "participant_joined_group_call",
        "untypedData": {
            "server_info_data": "XYZ123ABC",
            "group_call_type": "0",
            "joining_user": "******",
        },
        "class": "AdminTextMessage",
    }
    assert CallJoined(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_admin_message(session, data)
def test_people_added(session):
    data = {
        "addedParticipants": [
            {
                "fanoutPolicy": "IRIS_MESSAGE_QUEUE",
                "firstName": "Abc",
                "fullName": "Abc Def",
                "initialFolder": "FOLDER_INBOX",
                "initialFolderId": {"systemFolderId": "INBOX"},
                "isMessengerUser": False,
                "userFbId": "1234",
            }
        ],
        "irisSeqId": "11223344",
        "irisTags": ["DeltaParticipantsAddedToGroupThread", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "3456",
            "adminText": "You added Abc Def to the group.",
            "folderId": {"systemFolderId": "INBOX"},
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "1122334455",
            "skipBumpThread": False,
            "tags": [],
            "threadKey": {"threadFbId": "4321"},
            "threadReadStateEffect": "KEEP_AS_IS",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456", "4567"],
        "requestContext": {"apiArgs": {}},
        "tqSeqId": "1111",
        "class": "ParticipantsAddedToGroupThread",
    }
    assert PeopleAdded(
        author=User(session=session, id="3456"),
        thread=Group(session=session, id="4321"),
        added=[User(session=session, id="1234")],
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_delta(session, data)
def test_plan_deleted(session):
    guest_list = [
        {
            "guest_list_state": "GOING",
            "node": {
                "id": "1234"
            }
        },
        {
            "guest_list_state": "INVITED",
            "node": {
                "id": "3456"
            }
        },
        {
            "guest_list_state": "INVITED",
            "node": {
                "id": "2345"
            }
        },
    ]
    data = {
        "irisSeqId": "1111111",
        "irisTags": ["DeltaAdminTextMessage", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "1234",
            "adminText":
            "You deleted the plan A plan for Mon, 20 Jan at 15:30.",
            "folderId": {
                "systemFolderId": "INBOX"
            },
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": False,
            "tags": ["source:titan:web", "no_push"],
            "threadKey": {
                "threadFbId": "4321"
            },
            "threadReadStateEffect": "MARK_UNREAD",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456"],
        "requestContext": {
            "apiArgs": {}
        },
        "tqSeqId": "1111",
        "type": "lightweight_event_delete",
        "untypedData": {
            "event_end_time": "0",
            "event_timezone": "",
            "event_id": "112233",
            "event_type": "EVENT",
            "event_location_id": "2233445566",
            "latitude": "0",
            "event_title": "A plan",
            "event_time": "1600000000",
            "event_seconds_to_notify_before": "3600",
            "guest_state_list": _util.json_minimal(guest_list),
            "event_note": "",
            "longitude": "0",
        },
        "class": "AdminTextMessage",
    }
    assert PlanDeleted(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        plan=PlanData(
            session=session,
            id="112233",
            time=datetime.datetime(2020,
                                   9,
                                   13,
                                   12,
                                   26,
                                   40,
                                   tzinfo=datetime.timezone.utc),
            title="A plan",
            location_id="2233445566",
            author_id=None,
            guests={
                "1234": GuestStatus.GOING,
                "2345": GuestStatus.INVITED,
                "3456": GuestStatus.INVITED,
            },
        ),
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_admin_message(session, data)
def test_poll_answered(session):
    poll_data = {
        "id":
        "112233",
        "text":
        "A poll",
        "total_count":
        3,
        "viewer_has_voted":
        "true",
        "options": [
            {
                "id": "1002",
                "text": "Option B",
                "total_count": 2,
                "viewer_has_voted": "true",
                "voters": ["1234", "2345"],
            },
            {
                "id": "1003",
                "text": "Option C",
                "total_count": 1,
                "viewer_has_voted": "true",
                "voters": ["1234"],
            },
            {
                "id": "1001",
                "text": "Option A",
                "total_count": 0,
                "viewer_has_voted": "false",
                "voters": [],
            },
        ],
    }
    data = {
        "irisSeqId": "1111111",
        "irisTags": ["DeltaAdminTextMessage", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "1234",
            "adminText":
            'You changed your vote to "Option B" and 1 other option in the poll: A poll.',
            "folderId": {
                "systemFolderId": "INBOX"
            },
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": False,
            "tags": ["source:titan:web"],
            "threadKey": {
                "threadFbId": "4321"
            },
            "threadReadStateEffect": "MARK_UNREAD",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456"],
        "requestContext": {
            "apiArgs": {}
        },
        "tqSeqId": "1111",
        "type": "group_poll",
        "untypedData": {
            "added_option_ids": "[1002,1003]",
            "removed_option_ids": "[1001]",
            "question_json": _util.json_minimal(poll_data),
            "event_type": "update_vote",
            "question_id": "112233",
        },
        "class": "AdminTextMessage",
    }
    assert PollVoted(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        poll=Poll(
            session=session,
            id="112233",
            question="A poll",
            options=[
                PollOption(
                    id="1002",
                    text="Option B",
                    vote=True,
                    voters=["1234", "2345"],
                    votes_count=2,
                ),
                PollOption(
                    id="1003",
                    text="Option C",
                    vote=True,
                    voters=["1234"],
                    votes_count=1,
                ),
                PollOption(id="1001",
                           text="Option A",
                           vote=False,
                           voters=[],
                           votes_count=0),
            ],
            options_count=3,
        ),
        added_ids=["1002", "1003"],
        removed_ids=["1001"],
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_admin_message(session, data)
def test_group_call_ended(session):
    data = {
        "irisSeqId": "1111111",
        "irisTags": ["DeltaAdminTextMessage", "is_from_iris_fanout"],
        "messageMetadata": {
            "actorFbId": "1234",
            "adminText": "The call ended.",
            "folderId": {
                "systemFolderId": "INBOX"
            },
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": False,
            "tags": ["source:titan:web", "no_push"],
            "threadKey": {
                "threadFbId": "4321"
            },
            "threadReadStateEffect": "MARK_UNREAD",
            "timestamp": "1500000000000",
            "unsendType": "deny_log_message",
        },
        "participants": ["1234", "2345", "3456"],
        "requestContext": {
            "apiArgs": {}
        },
        "tqSeqId": "1111",
        "type": "messenger_call_log",
        "untypedData": {
            "call_capture_attachments": "",
            "caller_id": "1234",
            "conference_name": "MESSENGER:1234567890",
            "rating": "0",
            "messenger_call_instance_id": "1234567890",
            "video": "",
            "event": "group_call_ended",
            "server_info": "XYZ123ABC",
            "call_duration": "31",
            "callee_id": "0",
        },
        "class": "AdminTextMessage",
    }
    data2 = {
        "callState": "NO_ONGOING_CALL",
        "messageMetadata": {
            "actorFbId": "1234",
            "folderId": {
                "systemFolderId": "INBOX"
            },
            "messageId": "mid.$XYZ",
            "offlineThreadingId": "11223344556677889900",
            "skipBumpThread": False,
            "tags": [],
            "threadKey": {
                "threadFbId": "4321"
            },
            "threadReadStateEffect": "KEEP_AS_IS",
            "timestamp": "1500000000000",
        },
        "class": "RtcCallData",
    }
    assert CallEnded(
        author=User(session=session, id="1234"),
        thread=Group(session=session, id="4321"),
        duration=datetime.timedelta(seconds=31),
        at=datetime.datetime(2017, 7, 14, 2, 40, tzinfo=datetime.timezone.utc),
    ) == parse_admin_message(session, data)