-
-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathhandlers.py
826 lines (696 loc) · 37 KB
/
handlers.py
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
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
"""
В данном модуле написаны хэндлеры для разных эвентов.
"""
from __future__ import annotations
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from cardinal import Cardinal
from FunPayAPI.types import OrderShortcut, Order
from FunPayAPI import exceptions, utils as fp_utils
from FunPayAPI.updater.events import *
from tg_bot import utils, keyboards
from Utils import cardinal_tools
from locales.localizer import Localizer
from threading import Thread
import configparser
from datetime import datetime
import logging
import time
import re
LAST_STACK_ID = ""
MSG_LOG_LAST_STACK_ID = ""
logger = logging.getLogger("FPC.handlers")
localizer = Localizer()
_ = localizer.translate
ORDER_HTML_TEMPLATE = """<a href="https://funpay.com/orders/DELITEST/" class="tc-item">
<div class="tc-date" bis_skin_checked="1">
<div class="tc-date-time" bis_skin_checked="1">сегодня, $date</div>
<div class="tc-date-left" bis_skin_checked="1">только что</div>
</div>
<div class="tc-order" bis_skin_checked="1">#DELITEST</div>
<div class="order-desc" bis_skin_checked="1">
<div bis_skin_checked="1">$lot_name</div>
<div class="text-muted" bis_skin_checked="1">Автовыдача, Тест</div>
</div>
<div class="tc-user" bis_skin_checked="1">
<div class="media media-user offline" bis_skin_checked="1">
<div class="media-left" bis_skin_checked="1">
<div class="avatar-photo pseudo-a" tabindex="0" data-href="https://funpay.com/users/000000/" style="background-image: url(/img/layout/avatar.png);" bis_skin_checked="1"></div>
</div>
<div class="media-body" bis_skin_checked="1">
<div class="media-user-name" bis_skin_checked="1">
<span class="pseudo-a" tabindex="0" data-href="https://funpay.com/users/000000/">$username</span>
</div>
<div class="media-user-status" bis_skin_checked="1">был 1.000.000 лет назад</div>
</div>
</div>
</div>
<div class="tc-status text-primary" bis_skin_checked="1">Оплачен</div>
<div class="tc-price text-nowrap tc-seller-sum" bis_skin_checked="1">999999.0 <span class="unit">₽</span></div>
</a>"""
# INIT MESSAGE
def save_init_chats_handler(c: Cardinal, e: InitialChatEvent):
"""
Кэширует существующие чаты (чтобы не отправлять приветственные сообщения).
"""
if c.MAIN_CFG["Greetings"].getboolean("sendGreetings") and e.chat.id not in c.old_users:
c.old_users[e.chat.id] = int(time.time())
cardinal_tools.cache_old_users(c.old_users)
# NEW MESSAGE / LAST CHAT MESSAGE CHANGED
def old_log_msg_handler(c: Cardinal, e: LastChatMessageChangedEvent):
"""
Логирует полученное сообщение.
"""
if not c.old_mode_enabled:
return
text, chat_name, chat_id = str(e.chat), e.chat.name, e.chat.id
username = c.account.username if not e.chat.unread else e.chat.name
logger.info(_("log_new_msg", chat_name, chat_id))
for index, line in enumerate(text.split("\n")):
if not index:
logger.info(f"$MAGENTA└───> $YELLOW{username}: $CYAN{line}")
else:
logger.info(f" $CYAN{line}")
def log_msg_handler(c: Cardinal, e: NewMessageEvent):
global MSG_LOG_LAST_STACK_ID
if e.stack.id() == MSG_LOG_LAST_STACK_ID:
return
chat_name, chat_id = e.message.chat_name, e.message.chat_id
logger.info(_("log_new_msg", chat_name, chat_id))
for index, event in enumerate(e.stack.get_stack()):
username, text = event.message.author, event.message.text or event.message.image_link
for line_index, line in enumerate(text.split("\n")):
if not index and not line_index:
logger.info(f"$MAGENTA└───> $YELLOW{username}: $CYAN{line}")
elif not line_index:
logger.info(f" $YELLOW{username}: $CYAN{line}")
else:
logger.info(f" $CYAN{line}")
MSG_LOG_LAST_STACK_ID = e.stack.id()
def greetings_handler(c: Cardinal, e: NewMessageEvent | LastChatMessageChangedEvent):
"""
Отправляет приветственное сообщение.
"""
if not c.MAIN_CFG["Greetings"].getboolean("sendGreetings"):
return
if not c.old_mode_enabled:
if isinstance(e, LastChatMessageChangedEvent):
return
obj = e.message
chat_id, chat_name, mtype, its_me, badge = obj.chat_id, obj.chat_name, obj.type, obj.author_id == c.account.id, obj.badge
else:
obj = e.chat
chat_id, chat_name, mtype, its_me, badge = obj.id, obj.name, obj.last_message_type, not obj.unread, None
if any([time.time() - c.old_users.get(chat_id, 0) < float(
c.MAIN_CFG["Greetings"]["greetingsCooldown"]) * 24 * 60 * 60,
its_me, mtype in (MessageTypes.DEAR_VENDORS, MessageTypes.ORDER_CONFIRMED_BY_ADMIN), badge is not None,
(mtype is not MessageTypes.NON_SYSTEM and c.MAIN_CFG["Greetings"].getboolean("ignoreSystemMessages"))]):
return
logger.info(_("log_sending_greetings", chat_name, chat_id))
text = cardinal_tools.format_msg_text(c.MAIN_CFG["Greetings"]["greetingsText"], obj)
Thread(target=c.send_message, args=(chat_id, text, chat_name), daemon=True).start()
def add_old_user_handler(c: Cardinal, e: NewMessageEvent | LastChatMessageChangedEvent):
"""
Добавляет пользователя в список написавших.
"""
if not c.old_mode_enabled:
if isinstance(e, LastChatMessageChangedEvent):
return
chat_id, mtype = e.message.chat_id, e.message.type
else:
chat_id, mtype = e.chat.id, e.chat.last_message_type
if not c.MAIN_CFG["Greetings"].getboolean("sendGreetings") or mtype == MessageTypes.DEAR_VENDORS:
return
c.old_users[chat_id] = int(time.time())
cardinal_tools.cache_old_users(c.old_users)
def send_response_handler(c: Cardinal, e: NewMessageEvent | LastChatMessageChangedEvent):
"""
Проверяет, является ли сообщение командой, и если да, отправляет ответ на данную команду.
"""
if not c.autoresponse_enabled:
return
if not c.old_mode_enabled:
if isinstance(e, LastChatMessageChangedEvent):
return
obj, mtext = e.message, str(e.message)
chat_id, chat_name, username = e.message.chat_id, e.message.chat_name, e.message.author
else:
obj, mtext = e.chat, str(e.chat)
chat_id, chat_name, username = obj.id, obj.name, obj.name
mtext = mtext.replace("\n", "")
if any([c.bl_response_enabled and username in c.blacklist, (command := mtext.strip().lower()) not in c.AR_CFG]):
return
logger.info(_("log_new_cmd", command, chat_name, chat_id))
response_text = cardinal_tools.format_msg_text(c.AR_CFG[command]["response"], obj)
Thread(target=c.send_message, args=(chat_id, response_text, chat_name), daemon=True).start()
def old_send_new_msg_notification_handler(c: Cardinal, e: LastChatMessageChangedEvent):
if any([not c.old_mode_enabled, not c.telegram, not e.chat.unread,
c.bl_msg_notification_enabled and e.chat.name in c.blacklist,
e.chat.last_message_type is not MessageTypes.NON_SYSTEM, str(e.chat).strip().lower() in c.AR_CFG.sections(),
str(e.chat).startswith("!автовыдача")]):
return
user = e.chat.name
if user in c.blacklist:
user = f"🚷 {user}"
elif e.chat.last_by_bot:
user = f"🐦 {user}"
else:
user = f"👤 {user}"
text = f"<i><b>{user}: </b></i><code>{utils.escape(str(e.chat))}</code>"
kb = keyboards.reply(e.chat.id, e.chat.name, extend=True)
Thread(target=c.telegram.send_notification, args=(text, kb, utils.NotificationTypes.new_message),
daemon=True).start()
def send_new_msg_notification_handler(c: Cardinal, e: NewMessageEvent) -> None:
"""
Отправляет уведомление о новом сообщении в телеграм.
"""
global LAST_STACK_ID
if not c.telegram or e.stack.id() == LAST_STACK_ID:
return
LAST_STACK_ID = e.stack.id()
chat_id, chat_name = e.message.chat_id, e.message.chat_name
if c.bl_msg_notification_enabled and chat_name in c.blacklist:
return
events = []
nm, m, f, b = False, False, False, False
for i in e.stack.get_stack():
if i.message.author_id == 0:
if c.include_fp_msg_enabled:
events.append(i)
f = True
elif i.message.by_bot:
if c.include_bot_msg_enabled:
events.append(i)
b = True
elif i.message.author_id == c.account.id:
if c.include_my_msg_enabled:
events.append(i)
m = True
else:
events.append(i)
nm = True
if not events:
return
if [m, f, b, nm].count(True) == 1 and \
any([m and not c.only_my_msg_enabled, f and not c.only_fp_msg_enabled, b and not c.only_bot_msg_enabled]):
return
text = ""
last_message_author_id = -1
last_by_bot = False
last_badge = None
last_by_vertex = False
for i in events:
message_text = str(e.message)
if message_text.strip().lower() in c.AR_CFG.sections() and len(events) < 2:
return
elif message_text.startswith("!автовыдача") and len(events) < 2:
return
if i.message.author_id == last_message_author_id and i.message.by_bot == last_by_bot and \
i.message.badge == last_badge and i.message.by_vertex == last_by_vertex:
author = ""
elif i.message.author_id == c.account.id:
author = f"<i><b>🤖 {_('you')} (<i>FPC</i>):</b></i> " if i.message.by_bot else f"<i><b>🫵 {_('you')}:</b></i> "
if i.message.is_autoreply:
author = f"<i><b>📦 {_('you')} ({i.message.badge}):</b></i> "
elif i.message.author_id == 0:
author = f"<i><b>🔵 {i.message.author}: </b></i>"
elif i.message.is_employee:
author = f"<i><b>🆘 {i.message.author} ({i.message.badge}): </b></i>"
elif i.message.author == i.message.chat_name:
author = f"<i><b>👤 {i.message.author}: </b></i>"
if i.message.is_autoreply:
author = f"<i><b>🛍️ {i.message.author} ({i.message.badge}):</b></i> "
elif i.message.author in c.blacklist:
author = f"<i><b>🚷 {i.message.author}: </b></i>"
elif i.message.by_bot:
author = f"<i><b>🐦 {i.message.author}: </b></i>"
elif i.message.by_vertex:
author = f"<i><b>🐺 {i.message.author}: </b></i>"
else:
author = f"<i><b>🆘 {i.message.author} {_('support')}: </b></i>"
msg_text = f"<code>{utils.escape(i.message.text)}</code>" if i.message.text else \
f"<a href=\"{i.message.image_link}\">" \
f"{c.show_image_name and not (i.message.author_id == c.account.id and i.message.by_bot) and i.message.image_name or _('photo')}</a>"
text += f"{author}{msg_text}\n\n"
last_message_author_id = i.message.author_id
last_by_bot = i.message.by_bot
last_by_vertex = i.message.by_vertex
last_badge = i.message.badge
kb = keyboards.reply(chat_id, chat_name, extend=True)
Thread(target=c.telegram.send_notification, args=(text, kb, utils.NotificationTypes.new_message),
daemon=True).start()
def send_review_notification(c: Cardinal, order: Order, chat_id: int, reply_text: str | None):
if not c.telegram:
return
reply_text = _("ntfc_review_reply_text").format(utils.escape(reply_text)) if reply_text else ""
Thread(target=c.telegram.send_notification,
args=(_("ntfc_new_review").format('⭐' * order.review.stars, order.id, utils.escape(order.review.text),
reply_text),
keyboards.new_order(order.id, order.buyer_username, chat_id),
utils.NotificationTypes.review),
daemon=True).start()
def process_review_handler(c: Cardinal, e: NewMessageEvent | LastChatMessageChangedEvent):
if not c.old_mode_enabled:
if isinstance(e, LastChatMessageChangedEvent):
return
obj = e.message
message_type, its_me = obj.type, obj.i_am_buyer
message_text, chat_id = str(obj), obj.chat_id
else:
obj = e.chat
message_type, its_me = obj.last_message_type, f" {c.account.username} " in str(obj)
message_text, chat_id = str(obj), obj.id
if message_type not in [types.MessageTypes.NEW_FEEDBACK, types.MessageTypes.FEEDBACK_CHANGED] or its_me:
return
def send_reply():
try:
order = c.get_order_from_object(obj)
if order is None:
raise Exception("Не удалось получить объект заказа.") # locale
except:
logger.error(f"Не удалось получить информацию о заказе для сообщения: \"{message_text}\".") # locale
logger.debug("TRACEBACK", exc_info=True)
return
if not order.review or not order.review.stars:
return
logger.info(f"Изменен отзыв на заказ #{order.id}.") # locale
toggle = f"star{order.review.stars}Reply"
text = f"star{order.review.stars}ReplyText"
reply_text = None
if c.MAIN_CFG["ReviewReply"].getboolean(toggle) and c.MAIN_CFG["ReviewReply"].get(text):
try:
# Укорачиваем текст до 999 символов (оставляем 1 на спецсимвол), до 10 строк
def format_text4review(text_: str):
max_l = 999
text_ = text_[:max_l + 1]
if len(text_) > max_l:
ln = len(text_)
indexes = []
for char in (".", "!", "\n"):
index1 = text_.rfind(char)
indexes.extend([index1, text_[:index1].rfind(char)])
text_ = text_[:max(indexes, key=lambda x: (x < ln - 1, x))] + "🐦"
text_ = text_.strip()
while text_.count("\n") > 9 and text.count("\n\n") > 1:
# заменяем с конца все двойные переносы строк на одинарные, но оставляем как можно больше
# переносов строк и не менее одного двойного переноса
text_ = text_[::-1].replace("\n\n", "\n",
min([text_.count("\n\n") - 1, text_.count("\n") - 9]))[::-1]
if text_.count("\n") > 9:
text_ = text_[::-1].replace("\n", " ", text_.count("\n") - 9)[::-1]
return text_
reply_text = cardinal_tools.format_order_text(c.MAIN_CFG["ReviewReply"].get(text), order)
reply_text = format_text4review(reply_text)
c.account.send_review(order.id, reply_text)
except:
logger.error(f"Произошла ошибка при ответе на отзыв {order.id}.") # locale
logger.debug("TRACEBACK", exc_info=True)
send_review_notification(c, order, chat_id, reply_text)
Thread(target=send_reply, daemon=True).start()
def send_command_notification_handler(c: Cardinal, e: NewMessageEvent | LastChatMessageChangedEvent):
"""
Отправляет уведомление о введенной команде в телеграм.
"""
if not c.telegram:
return
if not c.old_mode_enabled:
if isinstance(e, LastChatMessageChangedEvent):
return
obj, message_text = e.message, str(e.message)
chat_id, chat_name, username = e.message.chat_id, e.message.chat_name, e.message.author
else:
obj, message_text = e.chat, str(e.chat)
chat_id, chat_name, username = obj.id, obj.name, obj.name if obj.unread else c.account.username
if c.bl_cmd_notification_enabled and username in c.blacklist:
return
command = message_text.strip().lower()
if command not in c.AR_CFG or not c.AR_CFG[command].getboolean("telegramNotification"):
return
if not c.AR_CFG[command].get("notificationText"):
text = f"🧑💻 Пользователь <b><i>{username}</i></b> ввел команду <code>{utils.escape(command)}</code>." # locale
else:
text = cardinal_tools.format_msg_text(c.AR_CFG[command]["notificationText"], obj)
Thread(target=c.telegram.send_notification, args=(text, keyboards.reply(chat_id, chat_name),
utils.NotificationTypes.command), daemon=True).start()
def test_auto_delivery_handler(c: Cardinal, e: NewMessageEvent | LastChatMessageChangedEvent):
"""
Выполняет тест автовыдачи.
"""
if not c.old_mode_enabled:
if isinstance(e, LastChatMessageChangedEvent):
return
obj, message_text, chat_name, chat_id = e.message, str(e.message), e.message.chat_name, e.message.chat_id
else:
obj, message_text, chat_name, chat_id = e.chat, str(e.chat), e.chat.name, e.chat.id
if not message_text.startswith("!автовыдача"):
return
split = message_text.split()
if len(split) < 2:
logger.warning("Одноразовый ключ автовыдачи не обнаружен.") # locale
return
key = split[1].strip()
if key not in c.delivery_tests:
logger.warning("Невалидный одноразовый ключ автовыдачи.") # locale
return
lot_name = c.delivery_tests[key]
del c.delivery_tests[key]
date = datetime.now()
date_text = date.strftime("%H:%M")
html = ORDER_HTML_TEMPLATE.replace("$username", chat_name).replace("$lot_name", lot_name).replace("$date",
date_text)
fake_order = OrderShortcut("ADTEST", lot_name, 0.0, Currency.UNKNOWN, chat_name, 000000, chat_id,
types.OrderStatuses.PAID,
date, "Авто-выдача, Тест", None, html)
fake_event = NewOrderEvent(e.runner_tag, fake_order)
c.run_handlers(c.new_order_handlers, (c, fake_event,))
def send_categories_raised_notification_handler(c: Cardinal, cat: types.Category, error_text: str = "") -> None:
"""
Отправляет уведомление о поднятии лотов в Telegram.
"""
if not c.telegram:
return
text = f"""⤴️<b><i>Поднял все лоты категории</i></b> <code>{cat.name}</code>\n<tg-spoiler>{error_text}</tg-spoiler>""" # locale
Thread(target=c.telegram.send_notification,
args=(text,),
kwargs={"notification_type": utils.NotificationTypes.lots_raise}, daemon=True).start()
# Изменен список ордеров (REGISTER_TO_ORDERS_LIST_CHANGED)
def get_lot_config_by_name(c: Cardinal, name: str) -> configparser.SectionProxy | None:
"""
Ищет секцию лота в конфиге автовыдачи.
:param c: объект кардинала.
:param name: название лота.
:return: секцию конфига или None.
"""
for i in c.AD_CFG.sections():
if i in name:
return c.AD_CFG[i]
return None
def check_products_amount(config_obj: configparser.SectionProxy) -> int:
file_name = config_obj.get("productsFileName")
if not file_name:
return 1
return cardinal_tools.count_products(f"storage/products/{file_name}")
def update_current_lots_handler(c: Cardinal, e: OrdersListChangedEvent):
logger.info("Получаю информацию о лотах...") # locale
attempts = 3
while attempts:
try:
c.curr_profile = c.account.get_user(c.account.id)
c.curr_profile_last_tag = e.runner_tag
break
except:
logger.error("Произошла ошибка при получении информации о лотах.") # locale
logger.debug("TRACEBACK", exc_info=True)
attempts -= 1
time.sleep(2)
else:
logger.error("Не удалось получить информацию о лотах: превышено кол-во попыток.") # locale
return
def update_profile_lots_handler(c: Cardinal, e: OrdersListChangedEvent):
"""Добавляет в c.profile лоты, которых не было раньше."""
if c.curr_profile_last_tag != e.runner_tag or c.profile_last_tag == e.runner_tag:
return
c.profile_last_tag = e.runner_tag
lots = c.curr_profile.get_sorted_lots(1)
profile_lots = c.profile.get_sorted_lots(1)
for lot_id, lot in lots.items():
if lot_id not in profile_lots.keys():
c.profile.add_lot(lot)
# Новый ордер (REGISTER_TO_NEW_ORDER)
def log_new_order_handler(c: Cardinal, e: NewOrderEvent, *args):
"""
Логирует новый заказ.
"""
logger.info(f"Новый заказ! ID: $YELLOW#{e.order.id}$RESET")
def setup_event_attributes_handler(c: Cardinal, e: NewOrderEvent, *args):
config_section_name = None
config_section_obj = None
for lot_name in c.AD_CFG:
if lot_name in e.order.description:
config_section_obj = c.AD_CFG[lot_name]
config_section_name = lot_name
break
attributes = {"config_section_name": config_section_name, "config_section_obj": config_section_obj,
"delivered": False, "delivery_text": None, "goods_delivered": 0, "goods_left": None,
"error": 0, "error_text": None}
for i in attributes:
setattr(e, i, attributes[i])
if config_section_obj is None:
logger.info("Лот не найден в конфиге авто-выдачи!") # todo
else:
logger.info("Лот найден в конфиге авто-выдачи!") # todo
def send_new_order_notification_handler(c: Cardinal, e: NewOrderEvent, *args):
"""
Отправляет уведомления о новом заказе в телеграм.
"""
if not c.telegram:
return
if e.order.buyer_username in c.blacklist and c.MAIN_CFG["BlockList"].getboolean("blockNewOrderNotification"):
return
if not (config_obj := getattr(e, "config_section_obj")):
delivery_info = _("ntfc_new_order_not_in_cfg")
else:
if not c.autodelivery_enabled:
delivery_info = _("ntfc_new_order_ad_disabled")
elif config_obj.getboolean("disable"):
delivery_info = _("ntfc_new_order_ad_disabled_for_lot")
elif c.bl_delivery_enabled and e.order.buyer_username in c.blacklist:
delivery_info = _("ntfc_new_order_user_blocked")
else:
delivery_info = _("ntfc_new_order_will_be_delivered")
text = _("ntfc_new_order", f"{utils.escape(e.order.description)}, {utils.escape(e.order.subcategory_name)}",
e.order.buyer_username, f"{e.order.price} {e.order.currency}", e.order.id, delivery_info)
chat_id = c.account.get_chat_by_name(e.order.buyer_username, True).id
keyboard = keyboards.new_order(e.order.id, e.order.buyer_username, chat_id)
Thread(target=c.telegram.send_notification, args=(text, keyboard, utils.NotificationTypes.new_order),
daemon=True).start()
def deliver_goods(c: Cardinal, e: NewOrderEvent, *args):
chat_id = c.account.get_chat_by_name(e.order.buyer_username).id
cfg_obj = getattr(e, "config_section_obj")
delivery_text = cardinal_tools.format_order_text(cfg_obj["response"], e.order)
amount, goods_left, products = 1, -1, []
try:
if file_name := cfg_obj.get("productsFileName"):
if c.multidelivery_enabled and not cfg_obj.getboolean("disableMultiDelivery"):
amount = e.order.amount if e.order.amount else 1
products, goods_left = cardinal_tools.get_products(f"storage/products/{file_name}", amount)
delivery_text = delivery_text.replace("$product", "\n".join(products).replace("\\n", "\n"))
except Exception as exc:
logger.error(
f"Произошла ошибка при получении товаров для заказа $YELLOW{e.order.id}: {str(exc)}$RESET") # locale
logger.debug("TRACEBACK", exc)
setattr(e, "error", 1)
setattr(e, "error_text",
f"Произошла ошибка при получении товаров для заказа {e.order.id}: {str(exc)}") # locale
return
result = c.send_message(chat_id, delivery_text, e.order.buyer_username)
if not result:
logger.error(f"Не удалось отправить товар для ордера $YELLOW{e.order.id}$RESET.") # locale
setattr(e, "error", 1)
setattr(e, "error_text", f"Не удалось отправить сообщение с товаром для заказа {e.order.id}.") # locale
if file_name and products:
cardinal_tools.add_products(f"storage/products/{file_name}", products, at_zero_position=True)
else:
logger.info(f"Товар для заказа {e.order.id} выдан.") # locale
setattr(e, "delivered", True)
setattr(e, "delivery_text", delivery_text)
setattr(e, "goods_delivered", amount)
setattr(e, "goods_left", goods_left)
def deliver_product_handler(c: Cardinal, e: NewOrderEvent, *args) -> None:
"""
Обертка для deliver_product(), обрабатывающая ошибки.
"""
if not c.MAIN_CFG["FunPay"].getboolean("autoDelivery"):
return
if e.order.buyer_username in c.blacklist and c.bl_delivery_enabled:
logger.info(f"Пользователь {e.order.buyer_username} находится в ЧС и включена блокировка автовыдачи. "
f"$YELLOW(ID: {e.order.id})$RESET") # locale
return
if (config_section_obj := getattr(e, "config_section_obj")) is None:
return
if config_section_obj.getboolean("disable"):
logger.info(f"Для лота \"{e.order.description}\" отключена автовыдача.") # locale
return
c.run_handlers(c.pre_delivery_handlers, (c, e))
deliver_goods(c, e, *args)
c.run_handlers(c.post_delivery_handlers, (c, e))
# REGISTER_TO_POST_DELIVERY
def send_delivery_notification_handler(c: Cardinal, e: NewOrderEvent):
"""
Отправляет уведомление в телеграм об отправке товара.
"""
if c.telegram is None:
return
if getattr(e, "error"):
text = f"""❌ <code>{getattr(e, "error_text")}</code>"""
else:
amount = "<b>∞</b>" if getattr(e, "goods_left") == -1 else f"<code>{getattr(e, 'goods_left')}</code>"
text = f"""✅ Успешно выдал товар для ордера <code>{e.order.id}</code>.\n
🛒 <b><i>Товар:</i></b>
<code>{utils.escape(getattr(e, "delivery_text"))}</code>\n
📋 <b><i>Осталось товаров: </i></b>{amount}""" # locale
Thread(target=c.telegram.send_notification, args=(text,),
kwargs={"notification_type": utils.NotificationTypes.delivery}, daemon=True).start()
def update_lot_state(cardinal: Cardinal, lot: types.LotShortcut, task: int) -> bool:
"""
Обновляет состояние лота
:param cardinal: объект Кардинала.
:param lot: объект лота.
:param task: -1 - деактивировать лот. 1 - активировать лот.
:return: результат выполнения.
"""
attempts = 3
while attempts:
try:
lot_fields = cardinal.account.get_lot_fields(lot.id)
if task == 1:
lot_fields.active = True
cardinal.account.save_lot(lot_fields)
logger.info(f"Восстановил лот $YELLOW{lot.description}$RESET.") # locale
elif task == -1:
lot_fields.active = False
cardinal.account.save_lot(lot_fields)
logger.info(f"Деактивировал лот $YELLOW{lot.description}$RESET.") # locale
return True
except Exception as e:
if isinstance(e, exceptions.RequestFailedError) and e.status_code == 404:
logger.error(f"Произошла ошибка при изменении состояния лота $YELLOW{lot.description}$RESET:" # locale
"лот не найден.")
return False
logger.error(f"Произошла ошибка при изменении состояния лота $YELLOW{lot.description}$RESET.") # locale
logger.debug("TRACEBACK", exc_info=True)
attempts -= 1
time.sleep(2)
logger.error(
f"Не удалось изменить состояние лота $YELLOW{lot.description}$RESET: превышено кол-во попыток.") # locale
return False
def update_lots_states(cardinal: Cardinal, event: NewOrderEvent):
if not any([cardinal.autorestore_enabled, cardinal.autodisable_enabled]):
return
if cardinal.curr_profile_last_tag != event.runner_tag or cardinal.last_state_change_tag == event.runner_tag:
return
lots = cardinal.curr_profile.get_sorted_lots(1)
deactivated = []
restored = []
for lot in cardinal.profile.get_lots():
# -1 - деактивировать
# 0 - ничего не делать
# 1 - восстановить
current_task = 0
config_obj = get_lot_config_by_name(cardinal, lot.description)
# Если лот уже деактивирован
if lot.id not in lots:
# и не найден в конфиге автовыдачи (глобальное автовосстановление включено)
if config_obj is None:
if cardinal.autorestore_enabled:
current_task = 1
# и найден в конфиге автовыдачи
else:
# и глобальное автовосстановление вкл. + не выключено в самом лоте в конфиге автовыдачи
if cardinal.autorestore_enabled and config_obj.get("disableAutoRestore") in ["0", None]:
# если глобальная автодеактивация выключена - восстанавливаем.
if not cardinal.autodisable_enabled:
current_task = 1
# если глобальная автодеактивация включена - восстанавливаем только если есть товары.
else:
if check_products_amount(config_obj):
current_task = 1
# Если же лот активен
else:
# и найден в конфиге автовыдачи
if config_obj:
products_count = check_products_amount(config_obj)
# и все условия выполнены: нет товаров + включено глобальная автодеактивация + она не выключена в
# самом лоте в конфига автовыдачи - отключаем.
if all((not products_count, cardinal.MAIN_CFG["FunPay"].getboolean("autoDisable"),
config_obj.get("disableAutoDisable") in ["0", None])):
current_task = -1
if current_task:
result = update_lot_state(cardinal, lot, current_task)
if result:
if current_task == -1:
deactivated.append(lot.description)
elif current_task == 1:
restored.append(lot.description)
time.sleep(0.5)
if deactivated:
lots = "\n".join(deactivated) # locale
text = f"""🔴 <b>Деактивировал лоты:</b>
<code>{lots}</code>"""
Thread(target=cardinal.telegram.send_notification, args=(text,),
kwargs={"notification_type": utils.NotificationTypes.lots_deactivate}, daemon=True).start()
if restored:
lots = "\n".join(restored) # locale
text = f"""🟢 <b>Активировал лоты:</b>
<code>{lots}</code>"""
Thread(target=cardinal.telegram.send_notification, args=(text,),
kwargs={"notification_type": utils.NotificationTypes.lots_restore}, daemon=True).start()
cardinal.last_state_change_tag = event.runner_tag
def update_lots_state_handler(cardinal: Cardinal, event: NewOrderEvent, *args):
Thread(target=update_lots_states, args=(cardinal, event), daemon=True).start()
# BIND_TO_ORDER_STATUS_CHANGED
def send_thank_u_message_handler(c: Cardinal, e: OrderStatusChangedEvent):
"""
Отправляет ответное сообщение на подтверждение заказа.
"""
if not c.MAIN_CFG["OrderConfirm"].getboolean("sendReply") or e.order.status is not types.OrderStatuses.CLOSED:
return
text = cardinal_tools.format_order_text(c.MAIN_CFG["OrderConfirm"]["replyText"], e.order)
chat = c.account.get_chat_by_name(e.order.buyer_username, True)
logger.info(f"Пользователь $YELLOW{e.order.buyer_username}$RESET подтвердил выполнение заказа " # locale
f"$YELLOW{e.order.id}.$RESET") # locale
logger.info(f"Отправляю ответное сообщение ...") # locale
Thread(target=c.send_message, args=(chat.id, text, e.order.buyer_username),
kwargs={'watermark': c.MAIN_CFG["OrderConfirm"].getboolean("watermark")}, daemon=True).start()
def send_order_confirmed_notification_handler(cardinal: Cardinal, event: OrderStatusChangedEvent):
"""
Отправляет уведомление о подтверждении заказа в Telegram.
"""
if not event.order.status == types.OrderStatuses.CLOSED:
return
chat = cardinal.account.get_chat_by_name(event.order.buyer_username, True)
Thread(target=cardinal.telegram.send_notification,
args=(
f"""🪙 Пользователь <a href="https://funpay.com/chat/?node={chat.id}">{event.order.buyer_username}</a> """
f"""подтвердил выполнение заказа <code>{event.order.id}</code>.""", # locale
keyboards.new_order(event.order.id, event.order.buyer_username, chat.id),
utils.NotificationTypes.order_confirmed),
daemon=True).start()
def send_bot_started_notification_handler(c: Cardinal, *args):
"""
Отправляет уведомление о запуске бота в телеграм.
"""
if c.telegram is None:
return
text = _("fpc_init", c.VERSION, c.account.username, c.account.id,
c.balance.total_rub, c.balance.total_usd, c.balance.total_eur, c.account.active_sales)
for i in c.telegram.init_messages:
try:
c.telegram.bot.edit_message_text(text, i[0], i[1])
except:
continue
BIND_TO_INIT_MESSAGE = [save_init_chats_handler]
BIND_TO_LAST_CHAT_MESSAGE_CHANGED = [old_log_msg_handler,
greetings_handler,
add_old_user_handler,
send_response_handler,
process_review_handler,
old_send_new_msg_notification_handler,
send_command_notification_handler,
test_auto_delivery_handler]
BIND_TO_NEW_MESSAGE = [log_msg_handler,
greetings_handler,
add_old_user_handler,
send_response_handler,
process_review_handler,
send_new_msg_notification_handler,
send_command_notification_handler,
test_auto_delivery_handler]
BIND_TO_POST_LOTS_RAISE = [send_categories_raised_notification_handler]
BIND_TO_ORDERS_LIST_CHANGED = [update_current_lots_handler, update_profile_lots_handler]
BIND_TO_NEW_ORDER = [log_new_order_handler, setup_event_attributes_handler,
send_new_order_notification_handler, deliver_product_handler,
update_lots_state_handler]
BIND_TO_ORDER_STATUS_CHANGED = [send_thank_u_message_handler, send_order_confirmed_notification_handler]
BIND_TO_POST_DELIVERY = [send_delivery_notification_handler]
BIND_TO_POST_START = [send_bot_started_notification_handler]