Compare commits

...

15 Commits

Author SHA1 Message Date
Josiah Baldwin
59fb1f104e Bumped version 2025-04-01 12:17:38 -07:00
Josiah Baldwin
9bd3e10ed7 Merge pull request #45 from HuFlungDu/fix/device-open-url
Fixed weird issues in device_open_url
Fix #40
2025-04-01 12:08:43 -07:00
Josiah Baldwin
28e1d94ab9 Fixed weird issues in device_open_url 2025-04-01 12:06:41 -07:00
Josiah Baldwin
51325a89d3 Merge pull request #44 from DaanSelen/simonfix2
Fix #41
2025-04-01 11:53:18 -07:00
Josiah Baldwin
97dff80222 Merge pull request #42 from DaanSelen/lastconlastaddr
Display lastaddr and lastconnected in device object.
2025-04-01 11:51:43 -07:00
Daan
8da445348b Fix a bug discovered by @si458 and also suggested to be fixed 2025-03-21 00:06:24 +01:00
Daan
ab1fba5cc1 Display lastaddr and lastconnected in device object. 2025-03-20 23:55:13 +01:00
Josiah Baldwin
34a80cdda7 Merge pull request #39 from HuFlungDu/feat/websockets15
Feat/websockets15

Implement #38
2025-02-17 12:56:29 -08:00
Josiah Baldwin
fcf523dd62 Updated to work with websockets 15 proxy handling 2025-02-17 12:54:50 -08:00
Josiah Baldwin
9a1311167d Added traceback printing for autoreconnect test in case the error type changes 2025-02-17 12:54:01 -08:00
Josiah Baldwin
c2319fcf29 Updated requirements to websockets 15. This breaks things. 2025-02-17 12:22:31 -08:00
Josiah Baldwin
4d1c25a35c Merge pull request #37 from HuFlungDu/hotfix/1.1.2
Hotfix/1.1.2

Fixes #35 

Update Cryptography version to fix SSL vulnerability.
2025-02-17 12:12:13 -08:00
Josiah Baldwin
e226fff8dd Merge pull request #36 from HuFlungDu/hotfix/1.1.2
Hotfix/1.1.2

Fixes #35 

Update Cryptography version to fix SSL vulnerability.
2025-02-17 12:11:08 -08:00
Josiah Baldwin
f8600b09fe Merge pull request #33 from HuFlungDu/hotfix/1.1.1
Hotfix/1.1.1

Fix #29
2025-02-05 12:37:35 -08:00
Josiah Baldwin
61eebf1532 Merge pull request #26 from HuFlungDu/release/1.1.0
1.1.0 Release > dev
2025-01-08 14:25:18 -08:00
10 changed files with 35 additions and 27 deletions

View File

@@ -2,6 +2,17 @@
Changelog Changelog
========= =========
version 1.2.0
=============
Bugs:
* Fixed agent sometimes being None causing an oxception
* Fixed bad code in device_open_url
Features:
* Changed websockets version to 15. This now uses the proxy implemention from that library, instead of the previous hack.
* Added lastaddr and lastconnect to list_devices API
version 1.1.2 version 1.1.2
============= =============
Bugs: Bugs:

View File

@@ -8,5 +8,5 @@ sphinx-toolbox>=2.16.0
cffi~=1.17.1 cffi~=1.17.1
cryptography~=44.0.1 cryptography~=44.0.1
pycparser~=2.22 pycparser~=2.22
websockets~=14.2 websockets~=15.0.0
enum_tools enum_tools

Binary file not shown.

View File

@@ -45,7 +45,7 @@ python_requires = >=3.8
install_requires = install_requires =
importlib-metadata importlib-metadata
cryptography~=44.0.1 cryptography~=44.0.1
websockets~=14.2 websockets~=15.0.0
python-socks[asyncio]~=2.5.3 python-socks[asyncio]~=2.5.3

View File

@@ -59,7 +59,7 @@ class Device(object):
def __init__(self, nodeid, session, agent=None, def __init__(self, nodeid, session, agent=None,
name=None, desc=None, description=None, name=None, desc=None, description=None,
tags=None, users=None, tags=None, users=None,
agct=None, created_at=None, agct=None, created_at=None,
rname=None, computer_name=None, icon=constants.Icon.desktop, rname=None, computer_name=None, icon=constants.Icon.desktop,
mesh=None, mtype=None, meshtype=None, groupname=None, meshname=None, mesh=None, mtype=None, meshtype=None, groupname=None, meshname=None,
domain=None, host=None, ip=None, conn=None, connected=None, domain=None, host=None, ip=None, conn=None, connected=None,
@@ -71,7 +71,7 @@ class Device(object):
if links is None: if links is None:
links = {} links = {}
self.links = links self.links = links
if ("ver" in agent): if agent and "ver" in agent:
agent = { agent = {
"version": agent["ver"], "version": agent["ver"],
"id": agent["id"], "id": agent["id"],

View File

@@ -144,7 +144,7 @@ class Session(object):
options["additional_headers"] = headers options["additional_headers"] = headers
async for websocket in util.proxy_connect(self.url, proxy_url=self._proxy, process_exception=util._process_websocket_exception, **options): async for websocket in websockets.asyncio.client.connect(self.url, proxy=self._proxy, process_exception=util._process_websocket_exception, **options):
self.alive = True self.alive = True
self._socket_open.set() self._socket_open.set()
try: try:
@@ -482,6 +482,10 @@ class Session(object):
for node in nodes: for node in nodes:
if node["node"].get("meshid", None): if node["node"].get("meshid", None):
node["node"]["mesh"] = mesh.Mesh(node["node"].get("meshid"), self) node["node"]["mesh"] = mesh.Mesh(node["node"].get("meshid"), self)
if "lastConnect" in node and isinstance(node["lastConnect"], dict):
node["node"]["lastconnect"] = node["lastConnect"].get("time")
node["node"]["lastaddr"] = node["lastConnect"].get("addr")
del node["lastConnect"]
details = {} details = {}
for key, val in node.items(): for key, val in node.items():
if key != "node": if key != "node":
@@ -1737,10 +1741,11 @@ class Session(object):
tasks.append(tg.create_task(asyncio.wait_for(_(), timeout=timeout))) tasks.append(tg.create_task(asyncio.wait_for(_(), timeout=timeout)))
tasks.append({ "action": 'msg', "type": 'openUrl', "nodeid": nodeid, "url": url }, "device_open_url", timeout=timeout) tasks.append({ "action": 'msg', "type": 'openUrl', "nodeid": nodeid, "url": url }, "device_open_url", timeout=timeout)
success = tasks[0].result()
res = tasks[1].result() res = tasks[1].result()
success = tasks[2].result()
if data.get("result", "ok").lower() != "ok": if res.get("result", "ok").lower() != "ok":
raise exceptions.ServerError(data["result"]) raise exceptions.ServerError(data["result"])
if not success: if not success:

View File

@@ -67,7 +67,7 @@ class Tunnel(object):
self.url = self._session.url.replace('/control.ashx', '/meshrelay.ashx?browser=1&p=' + str(self._protocol) + '&nodeid=' + self.node_id + '&id=' + self._tunnel_id + '&auth=' + self._authcookie["cookie"]) self.url = self._session.url.replace('/control.ashx', '/meshrelay.ashx?browser=1&p=' + str(self._protocol) + '&nodeid=' + self.node_id + '&id=' + self._tunnel_id + '&auth=' + self._authcookie["cookie"])
async for websocket in util.proxy_connect(self.url, proxy_url=self._session._proxy, process_exception=util._process_websocket_exception, **options): async for websocket in websockets.asyncio.client.connect(self.url, proxy=self._session._proxy, process_exception=util._process_websocket_exception, **options):
self.alive = True self.alive = True
self._socket_open.set() self._socket_open.set()
try: try:

View File

@@ -11,7 +11,6 @@ import ssl
import functools import functools
import urllib import urllib
import python_socks import python_socks
from python_socks.async_.asyncio import Proxy
from . import exceptions from . import exceptions
def _encode_cookie(o, key): def _encode_cookie(o, key):
@@ -164,17 +163,7 @@ def _process_websocket_exception(exc):
return exc return exc
if isinstance(exc, python_socks._errors.ProxyError): if isinstance(exc, python_socks._errors.ProxyError):
return None return None
return tmp # Proxy errors show up like this now, and it's default to error out. Handle explicitly.
if isinstance(exc, websockets.exceptions.InvalidProxyMessage):
class proxy_connect(websockets.asyncio.client.connect): return None
def __init__(self,*args, proxy_url=None, **kwargs): return tmp
self.proxy = None
if proxy_url is not None:
self.proxy = Proxy.from_url(proxy_url)
super().__init__(*args, **kwargs)
async def create_connection(self, *args, **kwargs):
if self.proxy is not None:
parsed = urllib.parse.urlparse(self.uri)
self.connection_kwargs["sock"] = await self.proxy.connect(dest_host=parsed.hostname, dest_port=parsed.port)
return await super().create_connection(*args, **kwargs)

View File

@@ -3,4 +3,4 @@ pytest-asyncio
cffi==1.17.1 cffi==1.17.1
cryptography~=44.0.1 cryptography~=44.0.1
pycparser==2.22 pycparser==2.22
websockets~=14.2 websockets~=15.0.0

View File

@@ -46,9 +46,11 @@ async def test_auto_reconnect(env):
for i in range(3): for i in range(3):
try: try:
await admin_session.ping(timeout=10) await admin_session.ping(timeout=10)
except: except* Exception as e:
continue print("".join(traceback.format_exception(e)))
break pass
else:
break
else: else:
raise Exception("Failed to reconnect") raise Exception("Failed to reconnect")
@@ -57,6 +59,7 @@ async def test_auto_reconnect(env):
try: try:
await admin_session.ping(timeout=10) await admin_session.ping(timeout=10)
except* Exception as e: except* Exception as e:
print("".join(traceback.format_exception(e)))
pass pass
else: else:
break break