-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathtg
More file actions
executable file
·299 lines (250 loc) · 9.29 KB
/
tg
File metadata and controls
executable file
·299 lines (250 loc) · 9.29 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
#!/usr/bin/env python3
"""
tg - A dead-simple Telegram CLI.
Usage:
tg send <chat> <message> Send a message to a chat
tg read <chat> [n] Read last n messages (default 10)
tg chats [n] List recent chats (default 20)
tg search <query> Search contacts/chats by name
tg file <chat> <path> Send a file
tg unread Show all unread messages
tg watch [interval] Poll for new messages (default 30s)
tg login Login / verify session
Chat can be a display name (partial match), phone number, or @username.
Setup:
1. Get api_id and api_hash from https://my.telegram.org
2. pip install telethon
3. Set env vars or create ~/.tg.conf:
TG_API_ID=12345
TG_API_HASH=abcdef...
4. tg login
"""
import sys
import os
import asyncio
from pathlib import Path
CONFIG_PATH = Path.home() / ".tg.conf"
SESSION_PATH = str(Path.home() / ".tg_session")
def load_config():
api_id = os.environ.get("TG_API_ID")
api_hash = os.environ.get("TG_API_HASH")
if not api_id and CONFIG_PATH.exists():
for line in CONFIG_PATH.read_text().splitlines():
line = line.strip()
if line.startswith("#") or "=" not in line:
continue
k, v = line.split("=", 1)
k, v = k.strip(), v.strip()
if k == "TG_API_ID":
api_id = v
elif k == "TG_API_HASH":
api_hash = v
elif k == "TG_SESSION":
global SESSION_PATH
SESSION_PATH = v
if not api_id or not api_hash:
print("Error: TG_API_ID and TG_API_HASH not set.")
print(f"Set env vars or create {CONFIG_PATH}:")
print(" TG_API_ID=12345")
print(" TG_API_HASH=abcdef...")
print("\nGet them from https://my.telegram.org")
sys.exit(1)
return int(api_id), api_hash
def print_usage():
print(__doc__.strip())
sys.exit(1)
async def get_client():
from telethon import TelegramClient
api_id, api_hash = load_config()
client = TelegramClient(SESSION_PATH, api_id, api_hash)
await client.start()
return client
async def resolve_chat(client, chat_str):
from telethon import errors
if chat_str.startswith("@"):
return await client.get_entity(chat_str)
if chat_str.startswith("+") or chat_str.isdigit():
try:
return await client.get_entity(chat_str)
except (ValueError, errors.UsernameNotOccupiedError):
pass
best = None
async for dialog in client.iter_dialogs(limit=200):
name = dialog.name or ""
if name == chat_str:
return dialog.entity
if chat_str.lower() in name.lower() and best is None:
best = dialog.entity
if best:
return best
return await client.get_entity(chat_str)
async def cmd_login():
client = await get_client()
me = await client.get_me()
print(f"Logged in as: {me.first_name} {me.last_name or ''} (@{me.username or 'N/A'})")
print(f"Phone: {me.phone}")
await client.disconnect()
async def cmd_send(chat_str, message):
client = await get_client()
try:
entity = await resolve_chat(client, chat_str)
from telethon.utils import get_display_name
name = get_display_name(entity)
await client.send_message(entity, message)
print(f"Sent to {name}: {message}")
finally:
await client.disconnect()
async def cmd_read(chat_str, n=10):
client = await get_client()
try:
entity = await resolve_chat(client, chat_str)
from telethon.utils import get_display_name
chat_name = get_display_name(entity)
print(f"--- {chat_name} (last {n}) ---\n")
messages = []
async for msg in client.iter_messages(entity, limit=n):
messages.append(msg)
for msg in reversed(messages):
sender = ""
if msg.sender:
sender = get_display_name(msg.sender)
time_str = msg.date.strftime("%m-%d %H:%M")
text = msg.text or "[media]"
print(f"[{time_str}] {sender}: {text}")
finally:
await client.disconnect()
async def cmd_chats(n=20):
client = await get_client()
try:
print("--- Recent chats ---\n")
count = 0
async for dialog in client.iter_dialogs(limit=n):
unread = f" ({dialog.unread_count} unread)" if dialog.unread_count else ""
last = ""
if dialog.message and dialog.message.text:
last = dialog.message.text[:40]
elif dialog.message:
last = "[media]"
print(f" {dialog.name}{unread}: {last}")
count += 1
print(f"\nTotal: {count}")
finally:
await client.disconnect()
async def cmd_search(query):
client = await get_client()
try:
print(f"--- Search: {query} ---\n")
found = 0
async for dialog in client.iter_dialogs(limit=200):
name = dialog.name or ""
if query.lower() in name.lower():
print(f" {name}")
found += 1
if not found:
print(" No matches found")
finally:
await client.disconnect()
async def cmd_file(chat_str, file_path):
client = await get_client()
try:
entity = await resolve_chat(client, chat_str)
from telethon.utils import get_display_name
name = get_display_name(entity)
if not os.path.exists(file_path):
print(f"Error: file not found: {file_path}")
sys.exit(1)
print(f"Sending {file_path} to {name}...")
await client.send_file(entity, file_path)
print("Done.")
finally:
await client.disconnect()
async def cmd_unread():
client = await get_client()
from telethon.utils import get_display_name
try:
total = 0
async for dialog in client.iter_dialogs(limit=100):
if dialog.unread_count > 0:
chat_name = dialog.name or "Unknown"
print(f"\n--- {chat_name} ({dialog.unread_count} unread) ---")
messages = []
async for msg in client.iter_messages(dialog.entity, limit=dialog.unread_count):
messages.append(msg)
for msg in reversed(messages):
sender = get_display_name(msg.sender) if msg.sender else ""
time_str = msg.date.strftime("%H:%M")
text = msg.text or "[media]"
print(f" [{time_str}] {sender}: {text}")
total += dialog.unread_count
if total == 0:
print("No unread messages.")
else:
print(f"\nTotal: {total} unread")
finally:
await client.disconnect()
async def cmd_watch(interval=30):
client = await get_client()
from telethon.utils import get_display_name
seen = set()
async for dialog in client.iter_dialogs(limit=100):
if dialog.unread_count > 0:
async for msg in client.iter_messages(dialog.entity, limit=dialog.unread_count):
seen.add(msg.id)
print(f"Watching for new messages (every {interval}s)... Ctrl+C to stop\n")
try:
while True:
await asyncio.sleep(interval)
async for dialog in client.iter_dialogs(limit=100):
if dialog.unread_count > 0:
chat_name = dialog.name or "Unknown"
async for msg in client.iter_messages(dialog.entity, limit=dialog.unread_count):
if msg.id not in seen:
seen.add(msg.id)
sender = get_display_name(msg.sender) if msg.sender else ""
time_str = msg.date.strftime("%H:%M")
text = msg.text or "[media]"
print(f"[{time_str}] {chat_name} | {sender}: {text}")
except KeyboardInterrupt:
print("\nStopped.")
finally:
await client.disconnect()
def main():
if len(sys.argv) < 2:
print_usage()
cmd = sys.argv[1]
if cmd == "login":
asyncio.run(cmd_login())
elif cmd == "send":
if len(sys.argv) < 4:
print("Usage: tg send <chat> <message>")
sys.exit(1)
asyncio.run(cmd_send(sys.argv[2], " ".join(sys.argv[3:])))
elif cmd == "read":
if len(sys.argv) < 3:
print("Usage: tg read <chat> [n]")
sys.exit(1)
n = int(sys.argv[3]) if len(sys.argv) > 3 else 10
asyncio.run(cmd_read(sys.argv[2], n))
elif cmd == "chats":
n = int(sys.argv[2]) if len(sys.argv) > 2 else 20
asyncio.run(cmd_chats(n))
elif cmd == "search":
if len(sys.argv) < 3:
print("Usage: tg search <query>")
sys.exit(1)
asyncio.run(cmd_search(sys.argv[2]))
elif cmd == "file":
if len(sys.argv) < 4:
print("Usage: tg file <chat> <path>")
sys.exit(1)
asyncio.run(cmd_file(sys.argv[2], sys.argv[3]))
elif cmd == "unread":
asyncio.run(cmd_unread())
elif cmd == "watch":
interval = int(sys.argv[2]) if len(sys.argv) > 2 else 30
asyncio.run(cmd_watch(interval))
else:
print(f"Unknown command: {cmd}")
print_usage()
if __name__ == "__main__":
main()