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.
@@ -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,3 @@
1
+ from ._client import solve, get_saved_messages, listen_and_save_telegram, send_to_saved
2
+
3
+ __all__ = ["solve", "get_saved_messages", "listen_and_save_telegram", "send_to_saved"]
@@ -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,5 @@
1
+
2
+ [dev]
3
+ pytest
4
+ build
5
+ twine
@@ -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*"]
@@ -0,0 +1,4 @@
1
+ [egg_info]
2
+ tag_build =
3
+ tag_date = 0
4
+