abuz-solver-client 0.1.0__tar.gz
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.
- abuz_solver_client-0.1.0/PKG-INFO +18 -0
- abuz_solver_client-0.1.0/abuz_solver/__init__.py +3 -0
- abuz_solver_client-0.1.0/abuz_solver/_client.py +148 -0
- abuz_solver_client-0.1.0/abuz_solver/_config.py +37 -0
- abuz_solver_client-0.1.0/abuz_solver_client.egg-info/PKG-INFO +18 -0
- abuz_solver_client-0.1.0/abuz_solver_client.egg-info/SOURCES.txt +9 -0
- abuz_solver_client-0.1.0/abuz_solver_client.egg-info/dependency_links.txt +1 -0
- abuz_solver_client-0.1.0/abuz_solver_client.egg-info/requires.txt +5 -0
- abuz_solver_client-0.1.0/abuz_solver_client.egg-info/top_level.txt +1 -0
- abuz_solver_client-0.1.0/pyproject.toml +28 -0
- abuz_solver_client-0.1.0/setup.cfg +4 -0
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
Metadata-Version: 2.4
|
|
2
|
+
Name: abuz-solver-client
|
|
3
|
+
Version: 0.1.0
|
|
4
|
+
Summary: Client for remote task solver (single solve function).
|
|
5
|
+
Author: Abuz
|
|
6
|
+
License-Expression: MIT
|
|
7
|
+
Keywords: solver,client
|
|
8
|
+
Classifier: Programming Language :: Python :: 3
|
|
9
|
+
Classifier: Programming Language :: Python :: 3.8
|
|
10
|
+
Classifier: Programming Language :: Python :: 3.9
|
|
11
|
+
Classifier: Programming Language :: Python :: 3.10
|
|
12
|
+
Classifier: Programming Language :: Python :: 3.11
|
|
13
|
+
Classifier: Programming Language :: Python :: 3.12
|
|
14
|
+
Requires-Python: >=3.8
|
|
15
|
+
Provides-Extra: dev
|
|
16
|
+
Requires-Dist: pytest; extra == "dev"
|
|
17
|
+
Requires-Dist: build; extra == "dev"
|
|
18
|
+
Requires-Dist: twine; extra == "dev"
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
|
|
3
|
+
import sys
|
|
4
|
+
from typing import List, Optional
|
|
5
|
+
|
|
6
|
+
from . import _config
|
|
7
|
+
|
|
8
|
+
_no_proxy_done = False
|
|
9
|
+
|
|
10
|
+
def _no_proxy():
|
|
11
|
+
global _no_proxy_done
|
|
12
|
+
if _no_proxy_done:
|
|
13
|
+
return
|
|
14
|
+
import urllib.request
|
|
15
|
+
opener = urllib.request.build_opener(urllib.request.ProxyHandler({}))
|
|
16
|
+
urllib.request.install_opener(opener)
|
|
17
|
+
_no_proxy_done = True
|
|
18
|
+
|
|
19
|
+
def _client_log(msg: str, level: str = "INFO"):
|
|
20
|
+
try:
|
|
21
|
+
import os
|
|
22
|
+
line = f"{msg}\n"
|
|
23
|
+
log_path = os.path.join(_config._config_dir(), "solver_client.log")
|
|
24
|
+
os.makedirs(_config._config_dir(), exist_ok=True)
|
|
25
|
+
with open(log_path, "a", encoding="utf-8") as f:
|
|
26
|
+
f.write(line)
|
|
27
|
+
except Exception:
|
|
28
|
+
pass
|
|
29
|
+
|
|
30
|
+
def _get_url():
|
|
31
|
+
cfg = _config.load()
|
|
32
|
+
if cfg and cfg.get("url"):
|
|
33
|
+
return cfg["url"]
|
|
34
|
+
print("First run: enter server URL.")
|
|
35
|
+
url = input("Server URL (e.g. http://your-server:8001): ").strip()
|
|
36
|
+
if not url:
|
|
37
|
+
raise RuntimeError("Server URL is required")
|
|
38
|
+
_config.save_url(url)
|
|
39
|
+
return url
|
|
40
|
+
|
|
41
|
+
def _request(base_url: str, path: str, method: str = "GET", data: Optional[dict] = None) -> dict:
|
|
42
|
+
_no_proxy()
|
|
43
|
+
import urllib.request
|
|
44
|
+
import urllib.error
|
|
45
|
+
import json
|
|
46
|
+
url = f"{base_url.rstrip('/')}{path}"
|
|
47
|
+
req = urllib.request.Request(url, method=method)
|
|
48
|
+
if data is not None:
|
|
49
|
+
req.data = json.dumps(data).encode("utf-8")
|
|
50
|
+
req.add_header("Content-Type", "application/json")
|
|
51
|
+
try:
|
|
52
|
+
with urllib.request.urlopen(req, timeout=600) as r:
|
|
53
|
+
return json.loads(r.read().decode("utf-8"))
|
|
54
|
+
except urllib.error.HTTPError as e:
|
|
55
|
+
body = e.read().decode("utf-8", errors="replace") if e.fp else ""
|
|
56
|
+
raise RuntimeError(f"HTTP {e.code}: {body[:200]}" if body else str(e)) from e
|
|
57
|
+
|
|
58
|
+
def solve(task: str, lang: str, out_file: Optional[str] = None, server_url: Optional[str] = None, out_dir: str = "solutions") -> str:
|
|
59
|
+
"""
|
|
60
|
+
Отправить задачу на сервер, вернуть решение (код). Результат всегда сохраняется у клиента.
|
|
61
|
+
task: текст задания, lang: язык (python, c++, ...).
|
|
62
|
+
out_file: дополнительно сохранить в этот файл.
|
|
63
|
+
out_dir: папка для автосохранения (по умолчанию solutions/); результат пишется в out_dir/solution_YYYYMMDD_HHMMSS.txt.
|
|
64
|
+
server_url: URL сервера; если не указан — берётся из конфига.
|
|
65
|
+
"""
|
|
66
|
+
import os
|
|
67
|
+
from datetime import datetime
|
|
68
|
+
base_url = (server_url or "").strip() or _get_url()
|
|
69
|
+
data = _request(base_url, "/solve", method="POST", data={"task": task, "lang": lang})
|
|
70
|
+
if not data.get("ok"):
|
|
71
|
+
raise RuntimeError(data.get("detail", "Solve failed"))
|
|
72
|
+
content = data.get("content", "")
|
|
73
|
+
os.makedirs(out_dir, exist_ok=True)
|
|
74
|
+
default_path = os.path.join(out_dir, datetime.now().strftime("solution_%Y%m%d_%H%M%S.txt"))
|
|
75
|
+
with open(default_path, "w", encoding="utf-8") as f:
|
|
76
|
+
f.write(content)
|
|
77
|
+
if out_file:
|
|
78
|
+
with open(out_file, "w", encoding="utf-8") as f:
|
|
79
|
+
f.write(content)
|
|
80
|
+
return content
|
|
81
|
+
|
|
82
|
+
def send_to_saved(text: str, server_url: Optional[str] = None, out_dir: str = "telegram_sent") -> None:
|
|
83
|
+
"""
|
|
84
|
+
Отправить текст в «Избранное» Telegram через сервер. Копия отправленного сохраняется у клиента.
|
|
85
|
+
server_url: URL сервера; если не указан — берётся из конфига.
|
|
86
|
+
out_dir: папка для сохранения копии отправленного (по умолчанию telegram_sent/).
|
|
87
|
+
"""
|
|
88
|
+
import os
|
|
89
|
+
from datetime import datetime
|
|
90
|
+
base_url = (server_url or "").strip() or _get_url()
|
|
91
|
+
data = _request(base_url, "/telegram_send", method="POST", data={"text": text})
|
|
92
|
+
if not data.get("ok"):
|
|
93
|
+
raise RuntimeError(data.get("detail", "Send failed"))
|
|
94
|
+
os.makedirs(out_dir, exist_ok=True)
|
|
95
|
+
path = os.path.join(out_dir, datetime.now().strftime("sent_%Y%m%d_%H%M%S.txt"))
|
|
96
|
+
with open(path, "w", encoding="utf-8") as f:
|
|
97
|
+
f.write(text)
|
|
98
|
+
|
|
99
|
+
|
|
100
|
+
def get_saved_messages(limit: int = 50, server_url: Optional[str] = None, out_dir: str = "telegram_saved") -> List[dict]:
|
|
101
|
+
"""
|
|
102
|
+
Последние сообщения из «Избранное» — запрос к серверу. Результат сохраняется у клиента в out_dir.
|
|
103
|
+
Возвращает [{"text": ..., "date": ..., "id": ...}, ...].
|
|
104
|
+
out_dir: папка для сохранения каждого сообщения в отдельный .txt (по умолчанию telegram_saved/).
|
|
105
|
+
server_url: URL сервера; если не указан — берётся из конфига.
|
|
106
|
+
"""
|
|
107
|
+
import os
|
|
108
|
+
base_url = (server_url or "").strip() or _get_url()
|
|
109
|
+
data = _request(base_url, f"/telegram_saved_messages?limit={limit}")
|
|
110
|
+
raw = data.get("messages", [])
|
|
111
|
+
msgs = [{"id": m.get("id", 0), "text": m.get("text", ""), "date": m.get("date", "")} for m in raw]
|
|
112
|
+
os.makedirs(out_dir, exist_ok=True)
|
|
113
|
+
for m in msgs:
|
|
114
|
+
date_part = (m.get("date", "") or "").replace(":", "-").replace("T", "_")[:19] or "no_date"
|
|
115
|
+
path = os.path.join(out_dir, f"{date_part}_{m.get('id', 0)}.txt")
|
|
116
|
+
with open(path, "w", encoding="utf-8") as f:
|
|
117
|
+
f.write(m.get("text", "") or "(медиа/пусто)")
|
|
118
|
+
return msgs
|
|
119
|
+
|
|
120
|
+
def listen_and_save_telegram(out_dir: str = "telegram_saved", server_url: Optional[str] = None) -> None:
|
|
121
|
+
"""
|
|
122
|
+
Опрашивает сервер и сохраняет каждое новое сообщение из «Избранное» у клиента в out_dir (отдельный .txt).
|
|
123
|
+
Работает до Ctrl+C.
|
|
124
|
+
out_dir: папка для сохранения (по умолчанию telegram_saved/). server_url: URL сервера.
|
|
125
|
+
"""
|
|
126
|
+
import os
|
|
127
|
+
import time
|
|
128
|
+
base_url = (server_url or "").strip() or _get_url()
|
|
129
|
+
out_dir = os.path.abspath(out_dir)
|
|
130
|
+
os.makedirs(out_dir, exist_ok=True)
|
|
131
|
+
last_id = 0
|
|
132
|
+
print(f"Сервер: {base_url}. Сообщения сохраняются в: {out_dir}")
|
|
133
|
+
print("Выход: Ctrl+C.\n")
|
|
134
|
+
while True:
|
|
135
|
+
try:
|
|
136
|
+
data = _request(base_url, f"/telegram_new?after_id={last_id}")
|
|
137
|
+
for m in data.get("messages", []):
|
|
138
|
+
mid = m.get("id", 0)
|
|
139
|
+
text = m.get("text", "") or "(медиа/пусто)"
|
|
140
|
+
date = m.get("date", "").replace(":", "-").replace("T", "_")[:19]
|
|
141
|
+
path = os.path.join(out_dir, f"{date}_{mid}.txt")
|
|
142
|
+
with open(path, "w", encoding="utf-8") as f:
|
|
143
|
+
f.write(text)
|
|
144
|
+
last_id = max(last_id, mid)
|
|
145
|
+
print(f" сохранено: {os.path.basename(path)}")
|
|
146
|
+
except Exception as e:
|
|
147
|
+
print(f" ошибка запроса: {e}")
|
|
148
|
+
time.sleep(3)
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import json
|
|
2
|
+
import os
|
|
3
|
+
|
|
4
|
+
def _config_dir():
|
|
5
|
+
if os.name == "nt":
|
|
6
|
+
base = os.environ.get("APPDATA") or os.path.expanduser("~")
|
|
7
|
+
else:
|
|
8
|
+
base = os.environ.get("XDG_CONFIG_HOME") or os.path.expanduser("~/.config")
|
|
9
|
+
return os.path.join(base, "abuz_solver")
|
|
10
|
+
|
|
11
|
+
def _config_path():
|
|
12
|
+
return os.path.join(_config_dir(), "config.json")
|
|
13
|
+
|
|
14
|
+
def load():
|
|
15
|
+
p = _config_path()
|
|
16
|
+
if not os.path.isfile(p):
|
|
17
|
+
return None
|
|
18
|
+
try:
|
|
19
|
+
with open(p, "r", encoding="utf-8") as f:
|
|
20
|
+
return json.load(f)
|
|
21
|
+
except Exception:
|
|
22
|
+
return None
|
|
23
|
+
|
|
24
|
+
def save_url(server_url: str):
|
|
25
|
+
d = _config_dir()
|
|
26
|
+
os.makedirs(d, exist_ok=True)
|
|
27
|
+
p = _config_path()
|
|
28
|
+
data = load() or {}
|
|
29
|
+
data["url"] = server_url.rstrip("/")
|
|
30
|
+
with open(p, "w", encoding="utf-8") as f:
|
|
31
|
+
json.dump(data, f)
|
|
32
|
+
try:
|
|
33
|
+
os.chmod(p, 0o600)
|
|
34
|
+
except Exception:
|
|
35
|
+
pass
|
|
36
|
+
|
|
37
|
+
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
Metadata-Version: 2.4
|
|
2
|
+
Name: abuz-solver-client
|
|
3
|
+
Version: 0.1.0
|
|
4
|
+
Summary: Client for remote task solver (single solve function).
|
|
5
|
+
Author: Abuz
|
|
6
|
+
License-Expression: MIT
|
|
7
|
+
Keywords: solver,client
|
|
8
|
+
Classifier: Programming Language :: Python :: 3
|
|
9
|
+
Classifier: Programming Language :: Python :: 3.8
|
|
10
|
+
Classifier: Programming Language :: Python :: 3.9
|
|
11
|
+
Classifier: Programming Language :: Python :: 3.10
|
|
12
|
+
Classifier: Programming Language :: Python :: 3.11
|
|
13
|
+
Classifier: Programming Language :: Python :: 3.12
|
|
14
|
+
Requires-Python: >=3.8
|
|
15
|
+
Provides-Extra: dev
|
|
16
|
+
Requires-Dist: pytest; extra == "dev"
|
|
17
|
+
Requires-Dist: build; extra == "dev"
|
|
18
|
+
Requires-Dist: twine; extra == "dev"
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
pyproject.toml
|
|
2
|
+
abuz_solver/__init__.py
|
|
3
|
+
abuz_solver/_client.py
|
|
4
|
+
abuz_solver/_config.py
|
|
5
|
+
abuz_solver_client.egg-info/PKG-INFO
|
|
6
|
+
abuz_solver_client.egg-info/SOURCES.txt
|
|
7
|
+
abuz_solver_client.egg-info/dependency_links.txt
|
|
8
|
+
abuz_solver_client.egg-info/requires.txt
|
|
9
|
+
abuz_solver_client.egg-info/top_level.txt
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
abuz_solver
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
[build-system]
|
|
2
|
+
requires = ["setuptools>=61", "wheel"]
|
|
3
|
+
build-backend = "setuptools.build_meta"
|
|
4
|
+
|
|
5
|
+
[project]
|
|
6
|
+
name = "abuz-solver-client"
|
|
7
|
+
version = "0.1.0"
|
|
8
|
+
description = "Client for remote task solver (single solve function)."
|
|
9
|
+
requires-python = ">=3.8"
|
|
10
|
+
license = "MIT"
|
|
11
|
+
authors = [{ name = "Abuz" }]
|
|
12
|
+
keywords = ["solver", "client"]
|
|
13
|
+
classifiers = [
|
|
14
|
+
"Programming Language :: Python :: 3",
|
|
15
|
+
"Programming Language :: Python :: 3.8",
|
|
16
|
+
"Programming Language :: Python :: 3.9",
|
|
17
|
+
"Programming Language :: Python :: 3.10",
|
|
18
|
+
"Programming Language :: Python :: 3.11",
|
|
19
|
+
"Programming Language :: Python :: 3.12",
|
|
20
|
+
]
|
|
21
|
+
dependencies = []
|
|
22
|
+
|
|
23
|
+
[project.optional-dependencies]
|
|
24
|
+
dev = ["pytest", "build", "twine"]
|
|
25
|
+
|
|
26
|
+
[tool.setuptools.packages.find]
|
|
27
|
+
where = ["."]
|
|
28
|
+
include = ["abuz_solver*"]
|