# SignalR core client
[![Donate](https://img.shields.io/badge/Donate-PayPal-green.svg?logo=paypal&style=flat-square)](https://www.paypal.me/mandrewcito/1)
![Pypi](https://img.shields.io/pypi/v/signalrcore.svg)
[![Downloads](https://pepy.tech/badge/signalrcore/month)](https://pepy.tech/project/signalrcore/month)
[![Downloads](https://pepy.tech/badge/signalrcore)](https://pepy.tech/project/signalrcore)
![Issues](https://img.shields.io/github/issues/mandrewcito/signalrcore.svg)
![Open issues](https://img.shields.io/github/issues-raw/mandrewcito/signalrcore.svg)
![codecov.io](https://codecov.io/github/mandrewcito/signalrcore/coverage.svg?branch=master)
![logo alt](https://raw.githubusercontent.com/mandrewcito/signalrcore/master/docs/img/logo_temp.128.svg.png)
# Links
* [Dev to posts with library examples and implementation](https://dev.to/mandrewcito/singlar-core-python-client-58e7)
* [Pypi](https://pypi.org/project/signalrcore/)
* [Wiki - This Doc](https://mandrewcito.github.io/signalrcore/)
# Develop
Test server will be avaiable in [here](https://github.com/mandrewcito/signalrcore-containertestservers) and docker compose is required.
```bash
git clone https://github.com/mandrewcito/signalrcore-containertestservers
cd signalrcore-containertestservers
docker-compose up
cd ../signalrcore
make tests
```
## Known Issues
Issues related with closing sockets are inherited from the websocket-client library. Due to these problems i can't update the library to versions higher than websocket-client 0.54.0.
I'm working to solve it but for now its patched (Error number 1. Raises an exception, and then exception is treated for prevent errors).
If I update the websocket library I fall into error number 2, on local machine I can't reproduce it but travis builds fail (sometimes and randomly :()
* [1. Closing socket error](https://github.com/slackapi/python-slackclient/issues/171)
* [2. Random errors closing socket](https://github.com/websocket-client/websocket-client/issues/449)
# A Tiny How To
## Connect to a server without auth
```python
hub_connection = HubConnectionBuilder()\
.with_url(server_url)\
.configure_logging(logging.DEBUG)\
.with_automatic_reconnect({
"type": "raw",
"keep_alive_interval": 10,
"reconnect_interval": 5,
"max_attempts": 5
}).build()
```
## Connect to a server with auth
login_function must provide auth token
```python
hub_connection = HubConnectionBuilder()\
.with_url(server_url,
options={
"access_token_factory": login_function,
"headers": {
"mycustomheader": "mycustomheadervalue"
}
})\
.configure_logging(logging.DEBUG)\
.with_automatic_reconnect({
"type": "raw",
"keep_alive_interval": 10,
"reconnect_interval": 5,
"max_attempts": 5
}).build()
```
### Unauthorized errors
A login function must provide an error controller if authorization fails. When connection starts, if authorization fails exception will be propagated.
```python
def login(self):
response = requests.post(
self.login_url,
json={
"username": self.email,
"password": self.password
},verify=False)
if response.status_code == 200:
return response.json()["token"]
raise requests.exceptions.ConnectionError()
hub_connection.start() # this code will raise requests.exceptions.ConnectionError() if auth fails
```
## Configure logging
```python
HubConnectionBuilder()\
.with_url(server_url,
.configure_logging(logging.DEBUG)
...
```
## Configure socket trace
```python
HubConnectionBuilder()\
.with_url(server_url,
.configure_logging(logging.DEBUG, socket_trace=True)
...
```
## Configure your own handler
```python
import logging
handler = logging.StreamHandler()
handler.setLevel(logging.DEBUG)
hub_connection = HubConnectionBuilder()\
.with_url(server_url, options={"verify_ssl": False}) \
.configure_logging(logging.DEBUG, socket_trace=True, handler=handler)
...
```
## Configuring reconnection
After reaching max_attempts an exeption will be thrown and on_disconnect event will be fired.
```python
hub_connection = HubConnectionBuilder()\
.with_url(server_url)\
...
.build()
```
## Configuring additional headers
```python
hub_connection = HubConnectionBuilder()\
.with_url(server_url,
options={
"headers": {
"mycustomheader": "mycustomheadervalue"
}
})
...
.build()
```
## Configuring additional querystring parameters
```python
server_url ="http.... /?myquerystringparam=134&foo=bar"
connection = HubConnectionBuilder()\
.with_url(server_url,
options={
})\
.build()
```
## Congfiguring skip negotiation
```python
hub_connection = HubConnectionBuilder() \
.with_url("ws://"+server_url, options={
"verify_ssl": False,
"skip_negotiation": False,
"headers": {
}
}) \
.configure_logging(logging.DEBUG, socket_trace=True, handler=handler) \
.build()
```
## Configuring ping(keep alive)
keep_alive_interval sets the seconds of ping message
```python
hub_connection = HubConnectionBuilder()\
.with_url(server_url)\
.configure_logging(logging.DEBUG)\
.with_automatic_reconnect({
"type": "raw",
"keep_alive_interval": 10,
"reconnect_interval": 5,
"max_attempts": 5
}).build()
```
## Configuring logging
```python
hub_connection = HubConnectionBuilder()\
.with_url(server_url)\
.configure_logging(logging.DEBUG)\
.with_automatic_reconnect({
"type": "raw",
"keep_alive_interval": 10,
"reconnect_interval": 5,
"max_attempts": 5
}).build()
```
## Configure messagepack
```python
from signalrcore.protocol.messagepack_protocol import MessagePackHubProtocol
HubConnectionBuilder()\
.with_url(self.server_url, options={"verify_ssl":False})\
...
.with_hub_protocol(MessagePackHubProtocol())\
...
.build()
```
## Events
### On Connect / On Disconnect
on_open - fires when connection is opened and ready to send messages
on_close - fires when connection is closed
```python
hub_connection.on_open(lambda: print("connection opened and handshake received ready to send messages"))
hub_connection.on_close(lambda: print("connection closed"))
```
### On Hub Error (Hub Exceptions ...)
```
hub_connection.on_error(lambda data: print(f"An exception was thrown closed{data.error}"))
```
### Register an operation
ReceiveMessage - signalr method
print - function that has as parameters args of signalr method
```python
hub_connection.on("ReceiveMessage", print)
```
## Sending messages
SendMessage - signalr method
username, message - parameters of signalrmethod
```python
hub_connection.send("SendMessage", [username, message])
```
## Sending messages with callback
SendMessage - signalr method
username, message - parameters of signalrmethod
```python
send_callback_received = threading.Lock()
send_callback_received.acquire()
self.connection.send(
"SendMessage", # Method
[self.username, self.message], # Params
lambda m: send_callback_received.release()) # Callback
if not send_callback_received.acquire(timeout=1):
raise ValueError("CALLBACK NOT RECEIVED")
```
## Requesting streaming (Server to client)
```python
hub_connection.stream(
"Counter",
[len(self.items), 500]).subscribe({
"next": self.on_next,
"complete": self.on_complete,
"error": self.on_error
})
```
## Client side Streaming
```python
from signalrcore.subject import Subject
subject = Subject()
# Start Streaming
hub_connection.send("UploadStream", subject)
# Each iteration
subject.next(str(iteration))
# End streaming
subject.complete()
```
# Full Examples
Examples will be avaiable [here](https://github.com/mandrewcito/signalrcore/tree/master/test/examples)
It were developed using package from [aspnet core - SignalRChat](https://codeload.github.com/aspnet/Docs/zip/master)
## Chat example
A mini example could be something like this:
```python
import logging
import sys
from signalrcore.hub_connection_builder import HubConnectionBuilder
def input_with_default(input_text, default_value):
value = input(input_text.format(default_value))
return default_value if value is None or value.strip() == "" else value
server_url = input_with_default('Enter your server url(default: {0}): ', "wss://localhost:44376/chatHub")
username = input_with_default('Enter your username (default: {0}): ', "mandrewcito")
handler = logging.StreamHandler()
handler.setLevel(logging.DEBUG)
hub_connection = HubConnectionBuilder()\
.with_url(server_url, options={"verify_ssl": False}) \
.configure_logging(logging.DEBUG, socket_trace=True, handler=handler) \
.with_automatic_reconnect({
"type": "interval",
"keep_alive_interval": 10,
"intervals": [1, 3, 5, 6, 7, 87, 3]
}).build()
hub_connection.on_open(lambda: print("connection opened and handshake received ready to send messages"))
hub_connection.on_close(lambda: print("connection closed"))
hub_connection.on("ReceiveMessage", print)
hub_connection.start()
message = None
# Do login
while message != "exit()":
message = input(">> ")
if message is not None and message != "" and message != "exit()":
hub_connection.send("SendMessage", [username, message])
hub_connection.stop()
sys.exit(0)
```
Raw data
{
"_id": null,
"home_page": "https://github.com/mandrewcito/signalrcore",
"name": "signalrcore-deng",
"maintainer": null,
"docs_url": null,
"requires_python": null,
"maintainer_email": null,
"keywords": "signalr core client 3.1",
"author": "mandrewcito",
"author_email": "anbaalo@gmail.com",
"download_url": "https://files.pythonhosted.org/packages/a7/7e/59115421bc87609b94def5d93f519a13c1c8b1d02668687602393969122e/signalrcore_deng-0.9.6.tar.gz",
"platform": null,
"description": "# SignalR core client\r\n[![Donate](https://img.shields.io/badge/Donate-PayPal-green.svg?logo=paypal&style=flat-square)](https://www.paypal.me/mandrewcito/1)\r\n![Pypi](https://img.shields.io/pypi/v/signalrcore.svg)\r\n[![Downloads](https://pepy.tech/badge/signalrcore/month)](https://pepy.tech/project/signalrcore/month)\r\n[![Downloads](https://pepy.tech/badge/signalrcore)](https://pepy.tech/project/signalrcore)\r\n![Issues](https://img.shields.io/github/issues/mandrewcito/signalrcore.svg)\r\n![Open issues](https://img.shields.io/github/issues-raw/mandrewcito/signalrcore.svg)\r\n![codecov.io](https://codecov.io/github/mandrewcito/signalrcore/coverage.svg?branch=master)\r\n\r\n![logo alt](https://raw.githubusercontent.com/mandrewcito/signalrcore/master/docs/img/logo_temp.128.svg.png)\r\n\r\n\r\n# Links \r\n\r\n* [Dev to posts with library examples and implementation](https://dev.to/mandrewcito/singlar-core-python-client-58e7)\r\n\r\n* [Pypi](https://pypi.org/project/signalrcore/)\r\n\r\n* [Wiki - This Doc](https://mandrewcito.github.io/signalrcore/)\r\n\r\n# Develop\r\n\r\nTest server will be avaiable in [here](https://github.com/mandrewcito/signalrcore-containertestservers) and docker compose is required.\r\n\r\n```bash\r\ngit clone https://github.com/mandrewcito/signalrcore-containertestservers\r\ncd signalrcore-containertestservers\r\ndocker-compose up\r\ncd ../signalrcore\r\nmake tests\r\n```\r\n\r\n## Known Issues\r\n\r\nIssues related with closing sockets are inherited from the websocket-client library. Due to these problems i can't update the library to versions higher than websocket-client 0.54.0. \r\nI'm working to solve it but for now its patched (Error number 1. Raises an exception, and then exception is treated for prevent errors). \r\nIf I update the websocket library I fall into error number 2, on local machine I can't reproduce it but travis builds fail (sometimes and randomly :()\r\n* [1. Closing socket error](https://github.com/slackapi/python-slackclient/issues/171)\r\n* [2. Random errors closing socket](https://github.com/websocket-client/websocket-client/issues/449)\r\n\r\n# A Tiny How To\r\n\r\n## Connect to a server without auth\r\n\r\n```python\r\nhub_connection = HubConnectionBuilder()\\\r\n .with_url(server_url)\\\r\n .configure_logging(logging.DEBUG)\\\r\n .with_automatic_reconnect({\r\n \"type\": \"raw\",\r\n \"keep_alive_interval\": 10,\r\n \"reconnect_interval\": 5,\r\n \"max_attempts\": 5\r\n }).build()\r\n```\r\n## Connect to a server with auth\r\n\r\nlogin_function must provide auth token\r\n\r\n```python\r\nhub_connection = HubConnectionBuilder()\\\r\n .with_url(server_url,\r\n options={\r\n \"access_token_factory\": login_function,\r\n \"headers\": {\r\n \"mycustomheader\": \"mycustomheadervalue\"\r\n }\r\n })\\\r\n .configure_logging(logging.DEBUG)\\\r\n .with_automatic_reconnect({\r\n \"type\": \"raw\",\r\n \"keep_alive_interval\": 10,\r\n \"reconnect_interval\": 5,\r\n \"max_attempts\": 5\r\n }).build()\r\n```\r\n### Unauthorized errors\r\nA login function must provide an error controller if authorization fails. When connection starts, if authorization fails exception will be propagated.\r\n\r\n```python\r\n def login(self):\r\n response = requests.post(\r\n self.login_url,\r\n json={\r\n \"username\": self.email,\r\n \"password\": self.password\r\n },verify=False)\r\n if response.status_code == 200:\r\n return response.json()[\"token\"]\r\n raise requests.exceptions.ConnectionError()\r\n\r\n hub_connection.start() # this code will raise requests.exceptions.ConnectionError() if auth fails\r\n```\r\n## Configure logging\r\n\r\n```python\r\nHubConnectionBuilder()\\\r\n .with_url(server_url,\r\n .configure_logging(logging.DEBUG)\r\n ...\r\n```\r\n## Configure socket trace\r\n```python \r\nHubConnectionBuilder()\\\r\n .with_url(server_url,\r\n .configure_logging(logging.DEBUG, socket_trace=True) \r\n ... \r\n ```\r\n## Configure your own handler\r\n```python\r\n import logging\r\nhandler = logging.StreamHandler()\r\nhandler.setLevel(logging.DEBUG)\r\nhub_connection = HubConnectionBuilder()\\\r\n .with_url(server_url, options={\"verify_ssl\": False}) \\\r\n .configure_logging(logging.DEBUG, socket_trace=True, handler=handler)\r\n ...\r\n ```\r\n## Configuring reconnection\r\nAfter reaching max_attempts an exeption will be thrown and on_disconnect event will be fired.\r\n```python\r\nhub_connection = HubConnectionBuilder()\\\r\n .with_url(server_url)\\\r\n ...\r\n .build()\r\n```\r\n## Configuring additional headers\r\n```python\r\nhub_connection = HubConnectionBuilder()\\\r\n .with_url(server_url,\r\n options={\r\n \"headers\": {\r\n \"mycustomheader\": \"mycustomheadervalue\"\r\n }\r\n })\r\n ...\r\n .build()\r\n```\r\n## Configuring additional querystring parameters\r\n```python\r\nserver_url =\"http.... /?myquerystringparam=134&foo=bar\"\r\nconnection = HubConnectionBuilder()\\\r\n .with_url(server_url,\r\n options={\r\n })\\\r\n .build()\r\n```\r\n## Congfiguring skip negotiation\r\n```python\r\nhub_connection = HubConnectionBuilder() \\\r\n .with_url(\"ws://\"+server_url, options={\r\n \"verify_ssl\": False,\r\n \"skip_negotiation\": False,\r\n \"headers\": {\r\n }\r\n }) \\\r\n .configure_logging(logging.DEBUG, socket_trace=True, handler=handler) \\\r\n .build()\r\n\r\n```\r\n## Configuring ping(keep alive)\r\n\r\nkeep_alive_interval sets the seconds of ping message\r\n\r\n```python\r\nhub_connection = HubConnectionBuilder()\\\r\n .with_url(server_url)\\\r\n .configure_logging(logging.DEBUG)\\\r\n .with_automatic_reconnect({\r\n \"type\": \"raw\",\r\n \"keep_alive_interval\": 10,\r\n \"reconnect_interval\": 5,\r\n \"max_attempts\": 5\r\n }).build()\r\n```\r\n## Configuring logging\r\n```python\r\nhub_connection = HubConnectionBuilder()\\\r\n .with_url(server_url)\\\r\n .configure_logging(logging.DEBUG)\\\r\n .with_automatic_reconnect({\r\n \"type\": \"raw\",\r\n \"keep_alive_interval\": 10,\r\n \"reconnect_interval\": 5,\r\n \"max_attempts\": 5\r\n }).build()\r\n```\r\n\r\n## Configure messagepack\r\n\r\n```python\r\nfrom signalrcore.protocol.messagepack_protocol import MessagePackHubProtocol\r\n\r\nHubConnectionBuilder()\\\r\n .with_url(self.server_url, options={\"verify_ssl\":False})\\\r\n ... \r\n .with_hub_protocol(MessagePackHubProtocol())\\\r\n ...\r\n .build()\r\n```\r\n## Events\r\n\r\n### On Connect / On Disconnect\r\non_open - fires when connection is opened and ready to send messages\r\non_close - fires when connection is closed\r\n```python\r\nhub_connection.on_open(lambda: print(\"connection opened and handshake received ready to send messages\"))\r\nhub_connection.on_close(lambda: print(\"connection closed\"))\r\n\r\n```\r\n### On Hub Error (Hub Exceptions ...)\r\n```\r\nhub_connection.on_error(lambda data: print(f\"An exception was thrown closed{data.error}\"))\r\n```\r\n### Register an operation \r\nReceiveMessage - signalr method\r\nprint - function that has as parameters args of signalr method\r\n```python\r\nhub_connection.on(\"ReceiveMessage\", print)\r\n```\r\n## Sending messages\r\nSendMessage - signalr method\r\nusername, message - parameters of signalrmethod\r\n```python\r\n hub_connection.send(\"SendMessage\", [username, message])\r\n```\r\n\r\n## Sending messages with callback\r\nSendMessage - signalr method\r\nusername, message - parameters of signalrmethod\r\n```python\r\n send_callback_received = threading.Lock()\r\n send_callback_received.acquire()\r\n self.connection.send(\r\n \"SendMessage\", # Method\r\n [self.username, self.message], # Params\r\n lambda m: send_callback_received.release()) # Callback\r\n if not send_callback_received.acquire(timeout=1):\r\n raise ValueError(\"CALLBACK NOT RECEIVED\")\r\n```\r\n\r\n## Requesting streaming (Server to client)\r\n```python\r\nhub_connection.stream(\r\n \"Counter\",\r\n [len(self.items), 500]).subscribe({\r\n \"next\": self.on_next,\r\n \"complete\": self.on_complete,\r\n \"error\": self.on_error\r\n })\r\n```\r\n## Client side Streaming\r\n```python\r\nfrom signalrcore.subject import Subject\r\n\r\nsubject = Subject()\r\n\r\n# Start Streaming\r\nhub_connection.send(\"UploadStream\", subject)\r\n\r\n# Each iteration\r\nsubject.next(str(iteration))\r\n\r\n# End streaming\r\nsubject.complete()\r\n```\r\n\r\n# Full Examples\r\n\r\nExamples will be avaiable [here](https://github.com/mandrewcito/signalrcore/tree/master/test/examples)\r\nIt were developed using package from [aspnet core - SignalRChat](https://codeload.github.com/aspnet/Docs/zip/master) \r\n\r\n## Chat example\r\nA mini example could be something like this:\r\n\r\n```python\r\nimport logging\r\nimport sys\r\nfrom signalrcore.hub_connection_builder import HubConnectionBuilder\r\n\r\n\r\ndef input_with_default(input_text, default_value):\r\n value = input(input_text.format(default_value))\r\n return default_value if value is None or value.strip() == \"\" else value\r\n\r\n\r\nserver_url = input_with_default('Enter your server url(default: {0}): ', \"wss://localhost:44376/chatHub\")\r\nusername = input_with_default('Enter your username (default: {0}): ', \"mandrewcito\")\r\nhandler = logging.StreamHandler()\r\nhandler.setLevel(logging.DEBUG)\r\nhub_connection = HubConnectionBuilder()\\\r\n .with_url(server_url, options={\"verify_ssl\": False}) \\\r\n .configure_logging(logging.DEBUG, socket_trace=True, handler=handler) \\\r\n .with_automatic_reconnect({\r\n \"type\": \"interval\",\r\n \"keep_alive_interval\": 10,\r\n \"intervals\": [1, 3, 5, 6, 7, 87, 3]\r\n }).build()\r\n\r\nhub_connection.on_open(lambda: print(\"connection opened and handshake received ready to send messages\"))\r\nhub_connection.on_close(lambda: print(\"connection closed\"))\r\n\r\nhub_connection.on(\"ReceiveMessage\", print)\r\nhub_connection.start()\r\nmessage = None\r\n\r\n# Do login\r\n\r\nwhile message != \"exit()\":\r\n message = input(\">> \")\r\n if message is not None and message != \"\" and message != \"exit()\":\r\n hub_connection.send(\"SendMessage\", [username, message])\r\n\r\nhub_connection.stop()\r\n\r\nsys.exit(0)\r\n\r\n```\r\n",
"bugtrack_url": null,
"license": null,
"summary": "\u4fee\u6539\u8bf4\u660e\uff1a1\u3001fork\u539f\u5e93\uff0c\u5728\u539f\u5e93\u7684\u57fa\u7840\u4e0a\u4fee\u590d\u4e86\u4e00\u4e9b\u591a\u7ebf\u7a0b\u76f8\u5173\u7684BUG\uff1b2\u3001\u652f\u6301\u8bbe\u7f6e\u7f51\u7edc\u4ee3\u7801\uff0c\u65b9\u4fbffiddler\u7b49\u8f6f\u4ef6\u6293\u5305\uff1b\uff08\u4ec5\u9879\u76ee\u81ea\u7528\uff0c\u5176\u4f59\u7528\u9014\u5efa\u8bae\u76f4\u63a5\u4f7f\u7528\u539f\u5e93signalrcore\u7684\u6700\u65b0\u7248\uff09",
"version": "0.9.6",
"project_urls": {
"Homepage": "https://github.com/mandrewcito/signalrcore"
},
"split_keywords": [
"signalr",
"core",
"client",
"3.1"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "1b6567c5502887522a707561466f76f07e764d870d4693a353366604dba93f6b",
"md5": "216ac02f7cb80eff15410897c141fce0",
"sha256": "08b7807b25dee961b6ab4dad53bd8ad30a55e04596a015fd59914683224a9e34"
},
"downloads": -1,
"filename": "signalrcore_deng-0.9.6-py3-none-any.whl",
"has_sig": false,
"md5_digest": "216ac02f7cb80eff15410897c141fce0",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": null,
"size": 36027,
"upload_time": "2024-10-14T02:39:48",
"upload_time_iso_8601": "2024-10-14T02:39:48.971415Z",
"url": "https://files.pythonhosted.org/packages/1b/65/67c5502887522a707561466f76f07e764d870d4693a353366604dba93f6b/signalrcore_deng-0.9.6-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "a77e59115421bc87609b94def5d93f519a13c1c8b1d02668687602393969122e",
"md5": "026e5ad84f0febc8aac6fb86d5ba223f",
"sha256": "3cf7ad0350e5cf202a60f28dacb2079233392f424b5b9a50edca3ba67bef71de"
},
"downloads": -1,
"filename": "signalrcore_deng-0.9.6.tar.gz",
"has_sig": false,
"md5_digest": "026e5ad84f0febc8aac6fb86d5ba223f",
"packagetype": "sdist",
"python_version": "source",
"requires_python": null,
"size": 25282,
"upload_time": "2024-10-14T02:39:50",
"upload_time_iso_8601": "2024-10-14T02:39:50.513596Z",
"url": "https://files.pythonhosted.org/packages/a7/7e/59115421bc87609b94def5d93f519a13c1c8b1d02668687602393969122e/signalrcore_deng-0.9.6.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-10-14 02:39:50",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "mandrewcito",
"github_project": "signalrcore",
"travis_ci": true,
"coveralls": false,
"github_actions": true,
"requirements": [
{
"name": "msgpack",
"specs": [
[
"==",
"1.0.2"
]
]
},
{
"name": "requests",
"specs": [
[
">=",
"2.22.0"
]
]
},
{
"name": "websocket-client",
"specs": [
[
"==",
"1.0.0"
]
]
}
],
"lcname": "signalrcore-deng"
}