Skip to content
This repository has been archived by the owner on Nov 2, 2023. It is now read-only.

Commit

Permalink
Merge pull request #21 from reinier-millo/develop
Browse files Browse the repository at this point in the history
Add keep-alive support for a more persistent connection

Websocket was closing after 30 seconds. WhatsApp don't use ping/pong timeout handled by websocket protocol. This timeouts must be disabled, and send custom message every 10 seconds to keep alive the connection. WhatsApp server will respond to the custom message with a plain text message with current server timestamps.
  • Loading branch information
reinier-millo authored Jan 30, 2022
2 parents 516a6ec + 0105b5d commit 3c8b404
Show file tree
Hide file tree
Showing 9 changed files with 494 additions and 36 deletions.
3 changes: 1 addition & 2 deletions .gitattributes
Original file line number Diff line number Diff line change
@@ -1,2 +1 @@
html/* linguist-vendored
html/kyros linguist-vendored
html/kyros/* linguist-vendored
61 changes: 38 additions & 23 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,10 @@ Kyros, for now, is a Python interface to communicate easier with WhatsApp Web AP
It provides an interface to connect and communicate with WhatsApp Web's websocket server.
Kyros will handle encryption and decryption kind of things.
In the future, Kyros is aimed to provide a full implementation of WhatsApp Web API which will give developers
a clean interface to work with (more or less like ![go-whatsapp](https://github.com/Rhymen/go-whatsapp)).
a clean interface to work with (more or less like [go-whatsapp](https://github.com/Rhymen/go-whatsapp)).
This module is designed to work with Python 3.6 or latest.
Special thanks to the creator of ![whatsapp-web-reveng](https://github.com/sigalor/whatsapp-web-reveng)
and ![go-whatsapp](https://github.com/Rhymen/go-whatsapp). This project is largely motivated by their work.
Special thanks to the creator of [whatsapp-web-reveng](https://github.com/sigalor/whatsapp-web-reveng)
and [go-whatsapp](https://github.com/Rhymen/go-whatsapp). This project is largely motivated by their work.
Please note that Kyros is not meant to be used actively in production servers as it is currently not
production ready. Use it at your own risk.

Expand All @@ -22,46 +22,61 @@ pip install git+https://[email protected]/ttycelery/kyros
```python
import asyncio
import logging
from os.path import exists

import pyqrcode

from kyros import Client, WebsocketMessage
import kyros

logging.basicConfig()
# set a logging level: just to know if something (bad?) happens
logging.getLogger("kyros").setLevel(logging.WARNING)
logger = logging.getLogger("kyros")
logger.setLevel(logging.DEBUG)

def handle_message(message):
logger.debug("Sample received message: %s", message)


async def main():
# create the Client instance using create class method
whatsapp = await kyros.Client.create()

# do a QR login
qr_data, scanned = await whatsapp.qr_login()

# generate qr code image
qr_code = pyqrcode.create(qr_data)
print(qr_code.terminal(quiet_zone=1))

try:
# wait for the QR code to be scanned
await scanned
except asyncio.TimeoutError:
# timed out (left unscanned), do a shutdown
await whatsapp.shutdown()
return

whatsapp = await kyros.Client.create(handle_message)

if exists("wp_session.json"):
currSession = kyros.Session.from_file("wp_session.json")
await whatsapp.restore_session(currSession)
else:
# do a QR login
qr_data, scanned = await whatsapp.qr_login()

# generate qr code image
qr_code = pyqrcode.create(qr_data)
print(qr_code.terminal(quiet_zone=1))
qr_code.svg('sample-qr.svg', scale=2)

try:
# wait for the QR code to be scanned
await scanned
except asyncio.TimeoutError:
# timed out (left unscanned), do a shutdown
await whatsapp.shutdown()
return

whatsapp.session.save_to_file("wp_session.json")

# how to send a websocket message
message = kyros.WebsocketMessage(None, ["query", "exist", "[email protected]"])
await whatsapp.websocket.send_message(message)

# receive a websocket message
print(await whatsapp.websocket.messages.get(message.tag))

# Await forever until app stopped with Ctrl+C
await asyncio.Future()

if __name__ == "__main__":
asyncio.run(main())
```
A "much more detailed documentation" kind of thing for this project is available ![here](https://ttycelery.github.io/kyros/).
A "much more detailed documentation" kind of thing for this project is available [here](https://ttycelery.github.io/kyros/).
You will see a piece of nightmare, happy exploring! Better documentation are being planned.

## Contribution
Expand Down
4 changes: 2 additions & 2 deletions dev-requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ pylint-django==2.0.12
pylint-flask==0.6
pylint-plugin-utils==0.6
PyQRCode==1.2.1
PyYAML==5.3.1
PyYAML==5.4
regex==2020.4.4
requirements-detector==0.6
rope==0.16.0
Expand All @@ -34,6 +34,6 @@ snowballstemmer==2.0.0
toml==0.10.0
typed-ast==1.4.1
websocket-client==0.57.0
websockets==8.1
websockets==9.1
wrapt==1.11.2
yapf==0.30.0
225 changes: 225 additions & 0 deletions kyros/bin_reader.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,225 @@
from .defines import WATags, WASingleByteTokens, WADoubleByteTokens, WAWebMessageInfo


class WABinaryReader:
"""WhatsApp Binary Reader
Read binary data from WhatsApp stream protocol
"""

def __init__(self, data):
self.data = data
self.index = 0

def check_eos(self, length):
"""Check if the end of the stream has been reached"""
if self.index + length > len(self.data):
raise EOFError("end of stream reached")

def read_byte(self):
"""Read single byte from the stream"""
self.check_eos(1)
ret = ord(chr(self.data[self.index]))
self.index += 1
return ret

def read_int_n(self, n, littleEndian=False):
"""Read integer value of n bytes"""
self.check_eos(n)
ret = 0
for i in range(n):
currShift = i if littleEndian else n - 1 - i
ret |= ord(chr(self.data[self.index + i])) << (currShift * 8)
self.index += n
return ret

def read_int16(self, littleEndian=False):
"""Read 16-bit integer value"""
return self.read_int_n(2, littleEndian)

def read_int20(self):
"""Read 20-bit integer value"""
self.check_eos(3)
ret = ((ord(chr(self.data[self.index])) & 15) << 16) + (ord(chr(self.data[self.index + 1])) << 8) + ord(chr(
self.data[self.index + 2]))
self.index += 3
return ret

def read_int32(self, littleEndian=False):
"""Read 32-bit integer value"""
return self.read_int_n(4, littleEndian)

def read_int64(self, littleEndian=False):
"""Read 64-bit integer value"""
return self.read_int_n(8, littleEndian)

def read_packed8(self, tag):
"""Read packed 8-bit string"""
startByte = self.read_byte()
ret = ""
for i in range(startByte & 127):
currByte = self.read_byte()
ret += self.unpack_byte(tag, (currByte & 0xF0)
>> 4) + self.unpack_byte(tag, currByte & 0x0F)
if (startByte >> 7) != 0:
ret = ret[:len(ret) - 1]
return ret

def unpack_byte(self, tag, value):
"""Handle byte as nibble digit or hex"""
if tag == WATags.NIBBLE_8:
return self.unpack_nibble(value)
elif tag == WATags.HEX_8:
return self.unpack_hex(value)

def unpack_nibble(self, value):
"""Convert value to digit or special chars"""
if 0 <= value <= 9:
return chr(ord('0') + value)
elif value == 10:
return "-"
elif value == 11:
return "."
elif value == 15:
return "\0"
raise ValueError("invalid nibble to unpack: " + value)

def unpack_hex(self, value):
"""Convert value to hex number"""
if value < 0 or value > 15:
raise ValueError("invalid hex to unpack: " + str(value))
if value < 10:
return chr(ord('0') + value)
else:
return chr(ord('A') + value - 10)

def is_list_tag(self, tag):
"""Check if the given tag is a list tag"""
return tag == WATags.LIST_EMPTY or tag == WATags.LIST_8 or tag == WATags.LIST_16

def read_list_size(self, tag):
"""Read the size of a list"""
if (tag == WATags.LIST_EMPTY):
return 0
elif (tag == WATags.LIST_8):
return self.read_byte()
elif (tag == WATags.LIST_16):
return self.read_int16()
raise ValueError("invalid tag for list size: " + str(tag))

def read_string(self, tag):
"""Read a string from the stream depending on the given tag"""
if tag >= 3 and tag <= 235:
token = self.get_token(tag)
if token == "s.whatsapp.net":
token = "c.us"
return token

if tag == WATags.DICTIONARY_0 or tag == WATags.DICTIONARY_1 or tag == WATags.DICTIONARY_2 or tag == WATags.DICTIONARY_3:
return self.get_token_double(tag - WATags.DICTIONARY_0, self.read_byte())
elif tag == WATags.LIST_EMPTY:
return
elif tag == WATags.BINARY_8:
return self.read_string_from_chars(self.read_byte())
elif tag == WATags.BINARY_20:
return self.read_string_from_chars(self.read_int20())
elif tag == WATags.BINARY_32:
return self.read_string_from_chars(self.read_int32())
elif tag == WATags.JID_PAIR:
i = self.read_string(self.read_byte())
j = self.read_string(self.read_byte())
if i is None or j is None:
raise ValueError("invalid jid pair: " + str(i) + ", " + str(j))
return i + "@" + j
elif tag == WATags.NIBBLE_8 or tag == WATags.HEX_8:
return self.read_packed8(tag)
else:
raise ValueError("invalid string with tag " + str(tag))

def read_string_from_chars(self, length):
"""Read indexed string from the stream with the given length"""
self.check_eos(length)
ret = self.data[self.index:self.index + length]
self.index += length
return ret

def read_attributes(self, n):
"""Read n data attributes"""
ret = {}
if n == 0:
return
for i in range(n):
index = self.read_string(self.read_byte())
ret[index] = self.read_string(self.read_byte())
return ret

def read_list(self, tag):
"""Read a list of data"""
ret = []
for i in range(self.read_list_size(tag)):
ret.append(self.read_node())
return ret

def read_node(self):
"""Read an information node"""
listSize = self.read_list_size(self.read_byte())
descrTag = self.read_byte()
if descrTag == WATags.STREAM_END:
raise ValueError("unexpected stream end")
descr = self.read_string(descrTag)
if listSize == 0 or not descr:
raise ValueError("invalid node")
attrs = self.read_attributes((listSize - 1) >> 1)
if listSize % 2 == 1:
return [descr, attrs, None]

tag = self.read_byte()
if self.is_list_tag(tag):
content = self.read_list(tag)
elif tag == WATags.BINARY_8:
content = self.read_bytes(self.read_byte())
elif tag == WATags.BINARY_20:
content = self.read_bytes(self.read_int20())
elif tag == WATags.BINARY_32:
content = self.read_bytes(self.read_int32())
else:
content = self.read_string(tag)
return [descr, attrs, content]

def read_bytes(self, n):
"""Read n bytes from the stream and return them as a string"""
ret = ""
for i in range(n):
ret += chr(self.read_byte())
return ret

def get_token(self, index):
"""Get the token at the given index."""
if index < 3 or index >= len(WASingleByteTokens):
raise ValueError("invalid token index: " + str(index))
return WASingleByteTokens[index]

def get_token_double(self, index1, index2):
"""Get a token from a double byte index"""
n = 256 * index1 + index2
if n < 0 or n >= len(WADoubleByteTokens):
raise ValueError("invalid token index: " + str(n))
return WADoubleByteTokens[n]


def read_message_array(msgs):
"""Read a list of messages"""
if not isinstance(msgs, list):
return msgs
ret = []
for x in msgs:
ret.append(WAWebMessageInfo.decode(bytes(x[2], "utf-8")) if isinstance(
x, list) and x[0] == "message" else x)
return ret


def read_binary(data, withMessages=False):
"""Read a binary message from WhatsApp stream"""
node = WABinaryReader(data).read_node()
if withMessages and node is not None and isinstance(node, list) and node[1] is not None:
node[2] = read_message_array(node[2])
return node
12 changes: 8 additions & 4 deletions kyros/client.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,23 +24,23 @@ class Client:
result in the failing of message delivery). A much better and pythonic
way to handle and raise exception is still a pending task."""
@classmethod
async def create(cls) -> Client:
async def create(cls, on_message=None) -> Client:
"""The proper way to instantiate `Client` class. Connects to
websocket server, also sets up the default client profile.
Returns a ready to use `Client` instance."""
instance = cls()
instance = cls(on_message)
await instance.setup_ws()
instance.load_profile(constants.CLIENT_VERSION,
constants.CLIENT_LONG_DESC,
constants.CLIENT_SHORT_DESC)
logger.info("Kyros instance created")
return instance

def __init__(self) -> None:
def __init__(self, on_message=None) -> None:
"""Initiate class. Do not initiate this way, use `Client.create()`
instead."""
self.profile = None
self.message_handler = message.MessageHandler()
self.message_handler = message.MessageHandler(on_message)
self.session = session.Session()
self.session.client_id = utilities.generate_client_id()
self.session.private_key = donna25519.PrivateKey()
Expand Down Expand Up @@ -125,6 +125,8 @@ async def wait_qr_scan():
self.session.enc_key = self.session.keys_decrypted[:32]
self.session.mac_key = self.session.keys_decrypted[32:64]

await self.websocket.keep_alive()

qr_fragments = [
self.session.server_id,
base64.b64encode(self.session.public_key.public).decode(),
Expand Down Expand Up @@ -184,6 +186,8 @@ async def restore():
self.session.server_token = info["serverToken"]

self.websocket.load_session(self.session) # reload references

await self.websocket.keep_alive()
return self.session

try:
Expand Down
Loading

0 comments on commit 3c8b404

Please sign in to comment.