From a0a03b61d0fb5263d47919c96d99f772cd057cd2 Mon Sep 17 00:00:00 2001 From: zayac Date: Sat, 20 Jan 2024 12:12:08 +0800 Subject: [PATCH] =?UTF-8?q?=E6=96=B0=E5=A2=9E=E4=BA=86=E4=B8=80=E4=BA=9B?= =?UTF-8?q?=E5=8A=9F=E8=83=BD=EF=BC=8C=E4=BF=AE=E5=A4=8D=E4=BA=86=E4=B8=80?= =?UTF-8?q?=E4=BA=9B=E5=B0=8F=E9=97=AE=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/bot/__init__.py | 0 src/bot/bot.py | 107 +++++++++++++++++++++++++++++++ src/change_url/__init__.py | 0 src/change_url/change_url.py | 82 +++++++++++++++++++++++ src/change_url/change_url_bot.py | 48 ++++++++++++++ src/change_url/hth.js | 17 +++++ src/change_url/ky.js | 20 ++++++ src/core/定时任务.py | 4 +- src/core/报数.py | 25 +++++++- src/core/查询存款失败用户.py | 25 +++++++- src/entity/user.py | 17 ++++- src/ui/app.py | 4 +- src/ui/data_query.py | 4 +- 13 files changed, 341 insertions(+), 12 deletions(-) create mode 100644 src/bot/__init__.py create mode 100644 src/bot/bot.py create mode 100644 src/change_url/__init__.py create mode 100644 src/change_url/change_url.py create mode 100644 src/change_url/change_url_bot.py create mode 100644 src/change_url/hth.js create mode 100644 src/change_url/ky.js diff --git a/src/bot/__init__.py b/src/bot/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/bot/bot.py b/src/bot/bot.py new file mode 100644 index 0000000..47e6b90 --- /dev/null +++ b/src/bot/bot.py @@ -0,0 +1,107 @@ +import schedule +import telebot +import time +import threading +from telebot import types + +from src.core.报数 import text_count_by_telegram_id, get_net_win_by_telegram_id +from src.core.查询存款失败用户 import get_pay_failed_by_telegram_id +from src.entity.user import get_all_users + +TOKEN = '6013830443:AAGzq1Tgtr_ZejU7bv0mab14xOwi0_64d0w' +bot = telebot.TeleBot(TOKEN) + + +def load_registered_users(): + """ + Load registered users from the data source. + """ + registered_users = set() + for user in get_all_users(): + ids = user.telegram_ids.split(',') + registered_users.update(map(int, ids)) + return registered_users + + +registered_users = load_registered_users() + + +def schedule_user_updates(): + """ + Schedule regular user updates. + """ + schedule.every(1).hour.do(lambda: load_registered_users()) + thread = threading.Thread(target=run_schedule) + thread.start() + + +def run_schedule(): + """ + Run scheduled tasks. + """ + while True: + schedule.run_pending() + time.sleep(1) + + +def is_registered(user_id): + """ + Check if a user is registered. + """ + return user_id in registered_users + + +def registered_user_only(func): + """ + Decorator to restrict access to registered users only. + """ + + def wrapper(message): + if not is_registered(message.from_user.id): + bot.send_message(message.chat.id, "您需要注册才能使用此功能") + return + return func(message) + + return wrapper + + +def create_menu(): + """ + Create a reply keyboard menu. + """ + markup = types.ReplyKeyboardMarkup(resize_keyboard=True, one_time_keyboard=True) + markup.row_width = 3 + markup.add(types.KeyboardButton("报数"), + types.KeyboardButton("负盈利"), + types.KeyboardButton("查失败")) + return markup + + +@bot.message_handler(commands=['start']) +@registered_user_only +def handle_start(message): + bot.send_message(message.chat.id, "菜单已生成:", reply_markup=create_menu()) + + +@bot.message_handler(commands=['userid']) +def user_id(message): + bot.reply_to(message, f"您的用户 ID 是: {message.from_user.id}") + + +@bot.message_handler(func=lambda message: True) +@registered_user_only +def handle_message(message): + # 获取用户id + telegram_id = message.from_user.id + response_mapping = { + "报数": text_count_by_telegram_id(telegram_id), + "负盈利": get_net_win_by_telegram_id(telegram_id), + "查失败": get_pay_failed_by_telegram_id(telegram_id) + } + + response = response_mapping.get(message.text, "未知选项") + bot.send_message(message.chat.id, response) + + +schedule_user_updates() # Schedule the user updates +bot.polling() # Start polling diff --git a/src/change_url/__init__.py b/src/change_url/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/change_url/change_url.py b/src/change_url/change_url.py new file mode 100644 index 0000000..eb369b3 --- /dev/null +++ b/src/change_url/change_url.py @@ -0,0 +1,82 @@ +import re + + +def extract_urls(text): + return re.findall(r'https?://\S+', text) + + +def read_file(filename): + try: + with open(filename, 'r') as file: + return file.read() + except IOError as e: + print(f"Error reading file {filename}: {e}") + raise + + +def write_file(filename, content): + try: + with open(filename, 'w') as file: + file.write(content) + except IOError as e: + print(f"Error writing to file {filename}: {e}") + raise + + +def replace_urls(js_content, url_mapping): + for placeholder, url in url_mapping.items(): + js_content = js_content.replace(placeholder, url) + return js_content + + +def get_hth_url_mapping(urls, is_seo): + if is_seo: + return { + "{{hthApp}}": urls[4], + "{{hthtyApp}}": urls[5], + "{{hthPc}}": urls[0], + "{{hthH5}}": urls[1] + } + else: + return { + "{{hthApp}}": urls[2], + "{{hthtyApp}}": urls[3], + "{{hthPc}}": urls[0], + "{{hthH5}}": urls[1] + } + + +def change_url(text): + try: + urls = extract_urls(text) + expected_filename = None + + if len(urls) >= 10: + js_content = read_file('ky.js.template') + ky_url_mapping = { + "{{kyApp1}}": urls[0], + "{{kyApp2}}": urls[1], + "{{kyPc1}}": urls[2], + "{{kyPc2}}": urls[3], + "{{kyH51}}": urls[4], + "{{kyH52}}": urls[5] + } + updated_content = replace_urls(js_content, ky_url_mapping) + expected_filename = 'ky.js' + write_file(expected_filename, updated_content) + elif len(urls) == 6: + js_content = read_file('hth.js.template') + is_seo = 'SEO' in text + hth_url_mapping = get_hth_url_mapping(urls, is_seo) + updated_content = replace_urls(js_content, hth_url_mapping) + expected_filename = 'hth.js' + write_file(expected_filename, updated_content) + else: + expected_filename = 'unknown.js' # 或者选择适合的默认文件名 + + # 如果一切顺利,返回 True 和预期的文件名 + return True, expected_filename + except Exception as e: + # 如果在过程中发生任何异常,打印错误并返回 False 和预期的文件名 + print(f"An error occurred: {e}") + return False, expected_filename if expected_filename else 'unknown.js' diff --git a/src/change_url/change_url_bot.py b/src/change_url/change_url_bot.py new file mode 100644 index 0000000..097ddca --- /dev/null +++ b/src/change_url/change_url_bot.py @@ -0,0 +1,48 @@ +import telebot +from loguru import logger + +from src.change_url.change_url import change_url + +TOKEN = '6013830443:AAGzq1Tgtr_ZejU7bv0mab14xOwi0_64d0w' +bot = telebot.TeleBot(TOKEN) +ky = ['技术客服域名值班'] +hth = ['YYZBH②拒绝私聊', '3-信息同步频道'] + + +@bot.message_handler(func=lambda message: message.forward_date is not None) +def handle_forwarded_message(message): + # if message.forward_from: + # # 直接获取转发来源用户信息 + # user = message.forward_from + # user_id = user.id + # user_first_name = user.first_name + # user_username = user.username if user.username else "No username" + # + # response = f"消息转发自 ID: {user_id}, 名字: {user_first_name}, 用户名: {user_username}" + # logger.debug(response) + # elif message.forward_from_chat: + # # 获取转发来源频道信息 + # channel = message.forward_from_chat + # channel_id = channel.id + # channel_title = channel.title + # + # response = f"消息转发自频道 ID: {channel_id}, 名称: {channel_title}" + # logger.debug(response) + # else: + # # 处理无法识别的转发 + # user_info = message.forward_origin + # response = f"这是一条转发的消息。{user_info}" + try: + res, filename = change_url(message.text) + if res: + msg = f'{filename}修改成功' + else: + msg = f'{filename}修改失败' + except Exception as e: + logger.error(f"An error occurred in change_url: {e}") + msg = "处理您的请求时发生错误。" + + bot.reply_to(message, msg) + + +bot.polling() diff --git a/src/change_url/hth.js b/src/change_url/hth.js new file mode 100644 index 0000000..dcf6d77 --- /dev/null +++ b/src/change_url/hth.js @@ -0,0 +1,17 @@ +const hthCode = '3016341' + +var hth_link = { + hthApp: '{{hthApp}}/?i_code='+hthCode, + hthtyApp: '{{hthtyApp}}/?i_code='+hthCode, + hthPc: '{{hthPc}}/register/?i_code='+hthCode, + hthH5: '{{hthH5}}/entry/register?i_code='+hthCode, +} + +function visit_hth(key) { + window['open'](hth_link[key]); + try { + LA.track(key); + } catch (error) { + console.error('An error occurred while tracking:', error); + } +} \ No newline at end of file diff --git a/src/change_url/ky.js b/src/change_url/ky.js new file mode 100644 index 0000000..e6a5358 --- /dev/null +++ b/src/change_url/ky.js @@ -0,0 +1,20 @@ +const kyCode = '97238304' + +var ky_link = { + kyApp1: '{{kyApp1}}/?i_code='+kyCode, + kyApp2: '{{kyApp2}}/?i_code='+kyCode, + kyPc1: '{{kyPc1}}/register/?i_code='+kyCode, + kyPc2: '{{kyPc2}}/register?i_code='+kyCode, + kyH51: '{{kyH51}}/entry/register?i_code='+kyCode, + kyH52: '{{kyH52}}/entry/register?i_code='+kyCode +} + + +function visit_ky(key) { + window['open'](ky_link[key]); + try { + LA.track(key); + } catch (error) { + console.error('An error occurred while tracking:', error); + } +} \ No newline at end of file diff --git a/src/core/定时任务.py b/src/core/定时任务.py index e9801b9..da54a61 100644 --- a/src/core/定时任务.py +++ b/src/core/定时任务.py @@ -2,7 +2,7 @@ import time import schedule from loguru import logger -from 报数 import get_net_win, text_count_by_user +from 报数 import get_net_win_by_user, text_count_by_user from 查询存款失败用户 import get_pay_failed_by_user from src.core.constant import BOT_TOKEN, COUNT_GROUP_ID @@ -27,7 +27,7 @@ def query_failed_deposit(username, password): def query_net_win(username, password) -> None: logger.info(f'Running query_net_win for username: {username}') user = get_user_by_username_and_password(username, password) - send_message(BOT_TOKEN, COUNT_GROUP_ID, get_net_win(user)) + send_message(BOT_TOKEN, COUNT_GROUP_ID, get_net_win_by_user(user)) logger.info(f'Finished query_net_win for username: {username}') diff --git a/src/core/报数.py b/src/core/报数.py index b6e4d80..7f7ea58 100644 --- a/src/core/报数.py +++ b/src/core/报数.py @@ -5,10 +5,11 @@ import pyperclip from src import logger from src.core import util +from src.core.util import get_curr_day from src.entity.database import db from src.entity.finance import Finance, get_finance -from src.entity.user import User -from src.entity.visual_list import VisualInfo, get_curr_data, get_visual_list +from src.entity.user import User, get_user_by_telegram_id +from src.entity.visual_list import VisualInfo, get_visual_list def get_statics(account, date=util.get_curr_day()) -> VisualInfo: @@ -38,6 +39,16 @@ def text_count_by_user(user: User, date: str) -> str: return text +def text_count_by_telegram_id(telegram_id: int) -> str: + visual_list = count_by_user(get_user_by_telegram_id(telegram_id), get_curr_day()) + text = '\n\n'.join( + f'{result.agentName}\n注册:{result.isNew}\n首存:{result.firstCount}\n日活:{int(result.countBets)}\n流水:{int(result.bets)}' + for result in visual_list + ) + logger.info(f'Generated text: {text}') + return text + + def get_finances_by_user(user: User, date) -> List[Finance]: accounts = user.accounts start_date = util.get_first_day_by_str(date) @@ -46,8 +57,16 @@ def get_finances_by_user(user: User, date) -> List[Finance]: return [future.result() for future in futures] -def get_net_win(user: User, date: str) -> str: +def get_net_win_by_user(user: User, date: str) -> str: finances = get_finances_by_user(user, date) finance_strings = [f"{finance.name}: {finance.netProfit}" for finance in finances] logger.info(f'Finance strings: {finance_strings}') return '\n'.join(finance_strings) + + +def get_net_win_by_telegram_id(telegram_id: int) -> str: + user = get_user_by_telegram_id(telegram_id) + finances = get_finances_by_user(user, get_curr_day()) + finance_strings = [f"{finance.name}: {finance.netProfit}" for finance in finances] + logger.info(f'Finance strings: {finance_strings}') + return '\n'.join(finance_strings) diff --git a/src/core/查询存款失败用户.py b/src/core/查询存款失败用户.py index b9fc7cc..3f2c799 100644 --- a/src/core/查询存款失败用户.py +++ b/src/core/查询存款失败用户.py @@ -5,10 +5,10 @@ from loguru import logger from src.core import api_request from src.core.constant import PAY_RECORD_LIST_URL -from src.core.util import get_first_day_month +from src.core.util import get_first_day_month, get_curr_day from src.entity.account import Account from src.entity.member import MemberList, get_member_list -from src.entity.user import User +from src.entity.user import User, get_user_by_telegram_id def get_pay_record_list(account: Account, date: str) -> Dict[str, List[str]]: @@ -80,3 +80,24 @@ def get_pay_failed_by_user(user: User, date: str) -> Optional[str]: logger.info(text) return text + + +def get_pay_failed_by_telegram_id(telegram_id: int) -> Optional[str]: + user = get_user_by_telegram_id(telegram_id) + with ThreadPoolExecutor(max_workers=len(user.accounts)) as executor: + futures = [executor.submit(get_pay_record_list, account, get_curr_day()) for account in user.accounts] + + # 使用列表推导式构建结果字符串 + text_lines = [ + "{}\n{}".format(res['name'], '\n'.join(res['names'])) + for future in futures if (res := future.result())['names'] + ] + + text = '\n'.join(text_lines) + + if not text: + logger.info('无存款失败用户') + return '无存款失败用户' + + logger.info(text) + return text diff --git a/src/entity/user.py b/src/entity/user.py index 03c4e9e..506e064 100644 --- a/src/entity/user.py +++ b/src/entity/user.py @@ -1,7 +1,8 @@ from dataclasses import dataclass from typing import List +from cachetools import cached, TTLCache -from sqlalchemy import String +from sqlalchemy import String, func from sqlalchemy.dialects.mssql import TINYINT from sqlalchemy.orm import Mapped, mapped_column, relationship @@ -32,3 +33,17 @@ def get_user_by_username_and_password(username: str, password: str) -> User: user = session.query(User).filter( User.username == username and User.password == password and User.status == 1).one() return user + + +def get_all_users(): + with db.Session() as session: + users = session.query(User).filter(User.status == 1).all() + return users + + +@cached(TTLCache(maxsize=100, ttl=3600)) +def get_user_by_telegram_id(telegram_id: int) -> User: + with db.Session() as session: + user = session.query(User).filter( + User.status == 1 and func.find_in_set(telegram_id, User.telegram_ids) > 0).one() + return user diff --git a/src/ui/app.py b/src/ui/app.py index 2ad74ac..544a87e 100644 --- a/src/ui/app.py +++ b/src/ui/app.py @@ -10,7 +10,7 @@ from PyQt6.QtWidgets import (QApplication, QCheckBox, QDateEdit, QHBoxLayout, QTabWidget, QTextEdit, QVBoxLayout, QWidget, QMessageBox, QSystemTrayIcon, QMenu) from src.core.message_client import send_message -from src.core.util import convert_data,resource_path +from src.core.util import convert_data, resource_path from src.entity.member import get_today_new_member_list from src.entity.pay_record import get_latest_deposit_user from src.entity.user import get_user_by_username_and_password @@ -343,7 +343,7 @@ class Application(QMainWindow): def handle_data_change(self, table, cell_data, col, account_username, notifications): if table.rowCount() > 1 and col != 0: - old_data_str = table.item(table.rowCount()-1, col).text() + old_data_str = table.item(table.rowCount() - 2, col).text() old_data = convert_data(old_data_str) new_data = convert_data(cell_data) diff --git a/src/ui/data_query.py b/src/ui/data_query.py index ed73b94..03f2ba8 100644 --- a/src/ui/data_query.py +++ b/src/ui/data_query.py @@ -3,7 +3,7 @@ import time from PyQt6.QtCore import QObject, QRunnable, QThread, pyqtSignal from src.core.salary import get_salary -from src.core.报数 import get_net_win, text_count_by_user +from src.core.报数 import get_net_win_by_user, text_count_by_user from src.core.查询存款失败用户 import get_pay_failed_by_user from src.entity.banner_info import get_banner_info @@ -54,7 +54,7 @@ class ButtonTask(QRunnable): auto_clipboard = True notify = True elif self.query_type == '负盈利': - result = get_net_win(self.user, self.selected_date_str) + result = get_net_win_by_user(self.user, self.selected_date_str) elif self.query_type == '薪资': result = get_salary(self.user, self.selected_date_str)