cycls 0.0.2.12__py3-none-any.whl → 0.0.2.13__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
cycls/cycls.py
CHANGED
|
@@ -1,27 +1,25 @@
|
|
|
1
1
|
from fastapi import FastAPI, Request
|
|
2
|
-
from fastapi.responses import StreamingResponse
|
|
2
|
+
from fastapi.responses import StreamingResponse
|
|
3
3
|
from pydantic import BaseModel
|
|
4
|
+
from typing import List, Dict, Optional
|
|
4
5
|
|
|
5
6
|
from functools import wraps
|
|
6
7
|
import uvicorn, socket, httpx
|
|
7
8
|
import inspect
|
|
8
9
|
|
|
9
|
-
|
|
10
|
-
|
|
10
|
+
import logging
|
|
11
|
+
logging.basicConfig(level=logging.ERROR)
|
|
11
12
|
|
|
12
13
|
import os
|
|
13
14
|
current_dir = os.path.dirname(os.path.abspath(__file__))
|
|
14
|
-
key_path = os.path.join(current_dir, '
|
|
15
|
-
# print(key_path)
|
|
15
|
+
key_path = os.path.join(current_dir, 'tuns')
|
|
16
16
|
|
|
17
|
-
from typing import List, Dict, Optional
|
|
18
17
|
class Message(BaseModel):
|
|
19
18
|
handle: str
|
|
20
19
|
content: str
|
|
21
|
-
|
|
20
|
+
id: str
|
|
22
21
|
history: Optional[List[Dict[str, str]]] = None
|
|
23
22
|
|
|
24
|
-
|
|
25
23
|
def find_available_port(start_port):
|
|
26
24
|
port = start_port
|
|
27
25
|
while True:
|
|
@@ -30,87 +28,89 @@ def find_available_port(start_port):
|
|
|
30
28
|
return port
|
|
31
29
|
port += 1
|
|
32
30
|
|
|
31
|
+
import asyncssh, asyncio
|
|
32
|
+
|
|
33
|
+
async def create_ssh_tunnel(x,y,z='tuns.sh'):
|
|
34
|
+
try:
|
|
35
|
+
async with asyncssh.connect(z,client_keys=[key_path],known_hosts=None) as conn:
|
|
36
|
+
listener = await conn.forward_remote_port(x, 80, 'localhost', y)
|
|
37
|
+
print("✦/✧","tunnel\n")
|
|
38
|
+
await listener.wait_closed()
|
|
39
|
+
except (OSError, asyncssh.Error) as e:
|
|
40
|
+
print("✦/✧",f"tunnel disconnected: {e}")
|
|
41
|
+
|
|
42
|
+
def register(handles, network, url, mode):
|
|
43
|
+
try:
|
|
44
|
+
with httpx.Client() as client:
|
|
45
|
+
response = client.post(f"{network}/register", json={"handles":handles, "url":url, "mode":mode})
|
|
46
|
+
if response.status_code==200:
|
|
47
|
+
data = (response.json()).get("content")
|
|
48
|
+
for i in data:print(f"✦/✧ {i[0]} / {network}/{i[1]}")
|
|
49
|
+
else:
|
|
50
|
+
print("✦/✧ failed to register ⚠️")
|
|
51
|
+
except Exception as e:
|
|
52
|
+
print(f"An error occurred: {e}")
|
|
53
|
+
|
|
54
|
+
async def run_server(x,y):
|
|
55
|
+
config = uvicorn.Config(x, host="127.0.0.1", port=y, log_level="error")
|
|
56
|
+
server = uvicorn.Server(config)
|
|
57
|
+
await server.serve()
|
|
58
|
+
|
|
33
59
|
class Cycls:
|
|
34
|
-
def __init__(self,network="https://cycls.com", port=find_available_port(8001)
|
|
35
|
-
|
|
60
|
+
def __init__(self, url="", network="https://cycls.com", port=find_available_port(8001)):
|
|
61
|
+
import uuid
|
|
62
|
+
self.subdomain = str(uuid.uuid4())[:8]
|
|
36
63
|
self.server = FastAPI()
|
|
37
64
|
self.network = network
|
|
38
65
|
self.port = port
|
|
39
66
|
self.url = url
|
|
40
|
-
self.
|
|
67
|
+
self.apps = {}
|
|
41
68
|
|
|
42
69
|
def __call__(self, handle):
|
|
43
|
-
self.handle = handle
|
|
44
70
|
def decorator(func):
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
def sync_wrapper(*args, **kwargs):
|
|
55
|
-
return func(*args, **kwargs)
|
|
56
|
-
self.server.post('/main')(sync_wrapper)
|
|
57
|
-
self.publish()
|
|
58
|
-
return sync_wrapper
|
|
71
|
+
@wraps(func)
|
|
72
|
+
async def async_wrapper(*args, **kwargs):
|
|
73
|
+
return await func(*args, **kwargs)
|
|
74
|
+
@wraps(func)
|
|
75
|
+
def sync_wrapper(*args, **kwargs):
|
|
76
|
+
return func(*args, **kwargs)
|
|
77
|
+
wrapper = async_wrapper if inspect.iscoroutinefunction(func) else sync_wrapper
|
|
78
|
+
self.apps["@"+handle] = wrapper
|
|
79
|
+
return wrapper
|
|
59
80
|
return decorator
|
|
60
|
-
|
|
61
|
-
def
|
|
81
|
+
|
|
82
|
+
async def gateway(self, request: Request):
|
|
83
|
+
data = await request.json()
|
|
84
|
+
handle = data.get('handle')
|
|
85
|
+
if handle in self.apps:
|
|
86
|
+
func = self.apps[handle]
|
|
87
|
+
message = Message(**data)
|
|
88
|
+
return await func(message) if inspect.iscoroutinefunction(func) else func(message)
|
|
89
|
+
return {"error": "Handle not found"}
|
|
90
|
+
|
|
91
|
+
def push(self):
|
|
92
|
+
self.server.post("/gateway")(self.gateway)
|
|
93
|
+
asyncio.run(self.publish())
|
|
94
|
+
|
|
95
|
+
async def publish(self):
|
|
62
96
|
prod=False
|
|
63
97
|
if self.url != "":
|
|
64
98
|
prod=True
|
|
65
99
|
|
|
66
|
-
|
|
100
|
+
print(f"✦/✧ serving at port: {self.port}")
|
|
67
101
|
if prod:
|
|
68
|
-
print("✦/✧","production mode
|
|
102
|
+
print("✦/✧",f"production mode | url: {self.url}")
|
|
103
|
+
register(list(self.apps.keys()), self.network, self.url+"/gateway", "prod")
|
|
69
104
|
else:
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
print("✦/✧",f"https://cycls.com/@{self.handle}")
|
|
76
|
-
print("")
|
|
77
|
-
|
|
78
|
-
with ThreadPoolExecutor() as executor:
|
|
79
|
-
if not prod:
|
|
80
|
-
self.register('dev')
|
|
81
|
-
executor.submit(self.tunnel)
|
|
82
|
-
else:
|
|
83
|
-
self.register('prod')
|
|
105
|
+
self.url = f"http://{self.subdomain}-cycls.tuns.sh"
|
|
106
|
+
print("✦/✧","development mode")
|
|
107
|
+
# print("✦/✧",f"url {self.url}")
|
|
108
|
+
register(list(self.apps.keys()), self.network, self.url+"/gateway", "dev")
|
|
109
|
+
t2 = asyncio.create_task(run_server(self.server,self.port))
|
|
84
110
|
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
else:
|
|
88
|
-
executor.submit(uvicorn.run(self.server, host="127.0.0.1", port=self.port, log_level="critical")) # perhaps keep traces?
|
|
89
|
-
|
|
90
|
-
def register(self, mode):
|
|
91
|
-
try:
|
|
92
|
-
with httpx.Client() as client:
|
|
93
|
-
response = client.post(f"{self.network}/register", json={"handle": f"@{self.handle}", "url": self.url, "mode": mode})
|
|
94
|
-
if response.status_code==200:
|
|
95
|
-
print(f"✦/✧ published 🎉")
|
|
96
|
-
print("")
|
|
97
|
-
else:
|
|
98
|
-
print("✦/✧ failed to register ⚠️") # exit app
|
|
99
|
-
except Exception as e:
|
|
100
|
-
print(f"An error occurred: {e}")
|
|
101
|
-
|
|
102
|
-
def tunnel(self):
|
|
103
|
-
# ssh_command = ['ssh', '-q', '-i', 'tuns', '-o', 'StrictHostKeyChecking=no', '-R', f'{self.handle}-cycls:80:localhost:{self.port}', 'tuns.sh']
|
|
104
|
-
# ssh_command = ['ssh', '-q', '-i', 'key.pub', '-o', 'StrictHostKeyChecking=no', '-R', f'{self.handle}-cycls:80:localhost:{self.port}', 'serveo.net']
|
|
105
|
-
ssh_command = ['ssh', '-q', '-i', key_path, '-o', 'StrictHostKeyChecking=no', '-R', f'{self.handle}-cycls:80:localhost:{self.port}', 'serveo.net']
|
|
106
|
-
try:
|
|
107
|
-
if self.debug:
|
|
108
|
-
process = subprocess.run(ssh_command,stdin=subprocess.DEVNULL) # very tricky! STDIN is what was messing with me
|
|
109
|
-
else:
|
|
110
|
-
process = subprocess.run(ssh_command, stdin=subprocess.DEVNULL, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
|
111
|
-
except Exception as e:
|
|
112
|
-
print(f"An error occurred: {e}") # exit app
|
|
111
|
+
t1 = asyncio.create_task(create_ssh_tunnel(f"{self.subdomain}-cycls", self.port))
|
|
112
|
+
await asyncio.gather(t1, t2) if not prod else asyncio.gather(t2)
|
|
113
113
|
|
|
114
114
|
Text = StreamingResponse
|
|
115
115
|
|
|
116
|
-
#
|
|
116
|
+
# poetry publish --build
|
cycls/tuns
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
-----BEGIN OPENSSH PRIVATE KEY-----
|
|
2
|
+
b3BlbnNzaC1rZXktdjEAAAAABG5vbmUAAAAEbm9uZQAAAAAAAAABAAABFwAAAAdzc2gtcn
|
|
3
|
+
NhAAAAAwEAAQAAAQEAxq9z8nrMUnCRBSDhOcUeAZ//dolIg7RQoXM+Yo6aPACuqRMjo70b
|
|
4
|
+
wzXotyIqFRHe5eLntoa5FLj4n/3/uKVjy+f0xwPxquzTRESJehmUNN7WXaXQt2TX7BmMeo
|
|
5
|
+
vxWiQ69qzcQMQx7FNJNKSsA4Ynw9fIg/NDvEAtnYotmcF37finCJzj3OOxAaAB53bAzfY2
|
|
6
|
+
GD0JL+W8bzuMXpyZaqD+33selRNLqWVbcgNrJW6IDtZXvT1gBRsVsycbNrBrwFuhafnLW1
|
|
7
|
+
KAP/ZLLRP4w+BUgJVJVu9PqTfTrAcnDwspduNJqAtYPMbvuoToip1kYwws3wl8hHpsal67
|
|
8
|
+
U5xZeOn+5QAAA9i3Fmc8txZnPAAAAAdzc2gtcnNhAAABAQDGr3PyesxScJEFIOE5xR4Bn/
|
|
9
|
+
92iUiDtFChcz5ijpo8AK6pEyOjvRvDNei3IioVEd7l4ue2hrkUuPif/f+4pWPL5/THA/Gq
|
|
10
|
+
7NNERIl6GZQ03tZdpdC3ZNfsGYx6i/FaJDr2rNxAxDHsU0k0pKwDhifD18iD80O8QC2dii
|
|
11
|
+
2ZwXft+KcInOPc47EBoAHndsDN9jYYPQkv5bxvO4xenJlqoP7fex6VE0upZVtyA2slbogO
|
|
12
|
+
1le9PWAFGxWzJxs2sGvAW6Fp+ctbUoA/9kstE/jD4FSAlUlW70+pN9OsBycPCyl240moC1
|
|
13
|
+
g8xu+6hOiKnWRjDCzfCXyEemxqXrtTnFl46f7lAAAAAwEAAQAAAQBFcWl7JMRpRALL4hQW
|
|
14
|
+
VvkH5F4rlgwMTGeqJld1pxXtRufFHHVmc2BSuHLgH0bKGnbnrokCWNAzl/r+II7SgKwCxs
|
|
15
|
+
3dCVncPe4RfEr4rBwK5p/SF3R9xPdbBAr/gg4XTXZ2ZTCOSoSQbwO1LKEakjcv0im5RLs1
|
|
16
|
+
/tBysasChIZgW9x4gGj7Kv4kz6Ojak/enlylwoz4q/dxaX9wV0op8YbnBubI3PS4gosYyz
|
|
17
|
+
fR9QAQbLrugo79xM9MpH1XjpQmk4UkuJX+OWqtZHq4Q2ZLuCoN4dv/mDcijAa/+lVtzerj
|
|
18
|
+
Ku6XtUREHRfdbKi88j+E7N4CfLW26UmV5xQhRjC8m+b5AAAAgQDoBpIUKoLEVVsWjc6n/e
|
|
19
|
+
TGip/qTlzcakODPELOzFK13GCAb/gj/sPxT6tUa382X9jcotgFSYBnO2TFA8gbBSXovWZv
|
|
20
|
+
zrXsNwfQBnvKih1Wngsq/bk7MYUvNPC0zomX0/dPi+riNb/5jLSEy5uICY4swMnsYb2Eyw
|
|
21
|
+
5CO+eHC9yiWAAAAIEA6Rx6TvBlMvqt5YbmdOX6LdNi9KNbh4ZmimFJu95IuvYUDol0aDa+
|
|
22
|
+
qc85GPBXt/vU6+mD3A1somE2WI2P0sswdYn+g7pbBdItA3NcCuqBFl735bOU6CQ5mygY2P
|
|
23
|
+
B/AE43+jeQUFfjBM+lDJNB8adffyBOQBhgmGAqGCq/p6GtgvMAAACBANoxpAHC8Q4cm4No
|
|
24
|
+
vsCefODzNUqgsOtgrJXfp3aDVx2pL9Ua71EAh2e5eWvr8YdvmhgU7ophAeEezQqt0nrQYW
|
|
25
|
+
xheJQQd1hgm50DY9ssKNcFHd8BkLwGVRbgDfwhPLDektZZqRlbrVsqIBkDurVI764h2v8M
|
|
26
|
+
G/fyU3Xv57Ed1vzHAAAAHm1vaGFtbWVkYWxydWpheWlATW9oYW1tZWRzLUFpcgECAwQ=
|
|
27
|
+
-----END OPENSSH PRIVATE KEY-----
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.1
|
|
2
2
|
Name: cycls
|
|
3
|
-
Version: 0.0.2.
|
|
3
|
+
Version: 0.0.2.13
|
|
4
4
|
Summary: Cycls SDK
|
|
5
5
|
Author: Mohammed Jamal
|
|
6
6
|
Author-email: mj@cycls.com
|
|
@@ -11,12 +11,22 @@ Classifier: Programming Language :: Python :: 3.9
|
|
|
11
11
|
Classifier: Programming Language :: Python :: 3.10
|
|
12
12
|
Classifier: Programming Language :: Python :: 3.11
|
|
13
13
|
Classifier: Programming Language :: Python :: 3.12
|
|
14
|
+
Requires-Dist: asyncssh (>=2.14.2,<3.0.0)
|
|
14
15
|
Requires-Dist: fastapi (>=0.111.0,<0.112.0)
|
|
15
16
|
Requires-Dist: httpx (>=0.27.0,<0.28.0)
|
|
16
17
|
Description-Content-Type: text/markdown
|
|
17
18
|
|
|
18
19
|
</br></br><p align="center"><img src="https://cycls.com/static/assets/favicon.svg" alt="Cycls"></p></br>
|
|
19
20
|
|
|
21
|
+
<div align="center">
|
|
22
|
+
<a href="https://pypi.org/project/cycls/" target="_blank" rel="noopener noreferrer">
|
|
23
|
+
<img loading="lazy" src="https://img.shields.io/pypi/v/cycls.svg" alt="PyPI" class="img_ev3q" style="display: inline;">
|
|
24
|
+
</a>
|
|
25
|
+
<a href="https://discord.gg/BMnaMatDC7" target="_blank" rel="noopener noreferrer">
|
|
26
|
+
<img loading="lazy" src="https://img.shields.io/discord/1175782747164389466" alt="Discord" class="img_ev3q" style="display: inline;">
|
|
27
|
+
</a>
|
|
28
|
+
</div>
|
|
29
|
+
|
|
20
30
|
# cycls.py
|
|
21
31
|
|
|
22
32
|
```sh
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
cycls/__init__.py,sha256=mBBqS9LS550Uqluyxs7VdD1W-9sDkQfwqeX681PNiis,39
|
|
2
|
+
cycls/cycls.py,sha256=L7JGqFarMNlqwbUbEIiUm0Tcf_eIddyRP-PzBdHZ2F0,4036
|
|
3
|
+
cycls/tuns,sha256=CT8olxDKOM0cY6r_bbSGtnWyEmEYFVDVHj9ug3gbYHk,1843
|
|
4
|
+
cycls-0.0.2.13.dist-info/METADATA,sha256=ACeVgl-K9Gn7YIn1g-JbNhOpVF516OxMJ2gOkXrloDU,3399
|
|
5
|
+
cycls-0.0.2.13.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
|
|
6
|
+
cycls-0.0.2.13.dist-info/RECORD,,
|
cycls/key.pub
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAACAQC022DIsZ11836rKBvsXcd1H7AMbirWyFSZczn3KkwaheLE90DigO/WkYhPMP5XJ9L5MIGM9nemiQhciZdN9f7z9o0xJh72ZI2+tl5Fa77a9r+IsuWbhn8fNGIkzvyYx8zwSTZV/f2nMYr4I7sjTVFVlH2Een6R8RCZpJPe9Q4ph+k2BCp4DeKzCuOrH6t//flm4vZnP8EXXXpSxic0TpbSL+Is7jeRsZdA4/7LrASCxzxnz4e0RCyy61B/TCkOHbSk06SPjoykA9amMMQ3T0OaO72bimiiCbRqEudgX0gNSVTxcRpmPARNJJDk72gAuOwAD3cpvIiW1nPu6Zhgb2VozQhsN+Rmh3ZjlYRvBVsHbwAkGb0iTBrtLA9ghiPLD2oXWdRI6P5XnNvSRyRXcrt59e1w9Z6XmKOmRl94NYciaMqH1qEEVnNCWUZyYKpMWRseL7v95FSNEoIL47+I4OPTF657sxZHlE85Oxei858kS6/b8GUT4yJgCq/PnhjN7UtGYCVRSv6UHGgXGvLMZAtHmbs3RlhEeLsluPvzgsDhmO6XzdKgh/MZIHKMIP/BEU11FT2lh/kB4EP48dqQPJMFNefB1ybEC509RCSdzaDATYIGRi7+FV4B1O15nC3EQTudFJl7io3chsLbkayAbGRInYVUJ0WDA5Et7ekIKRPdfw== m@m
|
cycls-0.0.2.12.dist-info/RECORD
DELETED
|
@@ -1,6 +0,0 @@
|
|
|
1
|
-
cycls/__init__.py,sha256=mBBqS9LS550Uqluyxs7VdD1W-9sDkQfwqeX681PNiis,39
|
|
2
|
-
cycls/cycls.py,sha256=G3KrB3-lO8poJTX0CVtf4BCPzsraUz7sGXiG-M9f6UQ,4366
|
|
3
|
-
cycls/key.pub,sha256=hWHddJ0TceUqDpm2xV4hDuNZKAz_wHpWox1yLaZ-JVw,729
|
|
4
|
-
cycls-0.0.2.12.dist-info/METADATA,sha256=MJqDNNHtJSmtg00gI5lrMl7UddPlKJ_HAIrTWlKBy-0,2864
|
|
5
|
-
cycls-0.0.2.12.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
|
|
6
|
-
cycls-0.0.2.12.dist-info/RECORD,,
|
|
File without changes
|