完成基础养号脚本
@@ -8,6 +8,7 @@ import wda
|
|||||||
from flask import Flask, request
|
from flask import Flask, request
|
||||||
from flask_cors import CORS
|
from flask_cors import CORS
|
||||||
from Entity.ResultData import ResultData
|
from Entity.ResultData import ResultData
|
||||||
|
from Utils.ThreadManager import ThreadManager
|
||||||
from script.ScriptManager import ScriptManager
|
from script.ScriptManager import ScriptManager
|
||||||
|
|
||||||
app = Flask(__name__)
|
app = Flask(__name__)
|
||||||
@@ -170,9 +171,29 @@ def growAccount():
|
|||||||
body = request.get_json()
|
body = request.get_json()
|
||||||
udid = body.get("udid")
|
udid = body.get("udid")
|
||||||
|
|
||||||
|
manager = ScriptManager()
|
||||||
|
event = threading.Event()
|
||||||
# 启动脚本
|
# 启动脚本
|
||||||
threading.Thread(target=ScriptManager.growAccount, args=(udid,)).start()
|
thread = threading.Thread(target=manager.growAccount, args=(udid,event))
|
||||||
|
thread.start()
|
||||||
|
# 添加到线程管理
|
||||||
|
ThreadManager.add(udid, thread, event)
|
||||||
return ResultData(data="").toJson()
|
return ResultData(data="").toJson()
|
||||||
|
|
||||||
|
# 停止脚本
|
||||||
|
@app.route("/stopScript", methods=['POST'])
|
||||||
|
def stopScript():
|
||||||
|
body = request.get_json()
|
||||||
|
udid = body.get("udid")
|
||||||
|
ThreadManager.stop(udid)
|
||||||
|
return ResultData(data="").toJson()
|
||||||
|
|
||||||
|
# AI聊天
|
||||||
|
@app.route('/aiChat', methods=['POST'])
|
||||||
|
def aiChat():
|
||||||
|
body = request.get_json()
|
||||||
|
udid = body.get("udid")
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
app.run("0.0.0.0", port=5000, debug=True, use_reloader=False)
|
app.run("0.0.0.0", port=5000, debug=True, use_reloader=False)
|
||||||
|
|||||||
@@ -74,7 +74,7 @@ class Deviceinfo(object):
|
|||||||
d = wda.USBClient(identifier, 8100)
|
d = wda.USBClient(identifier, 8100)
|
||||||
d.app_start(WdaAppBundleId)
|
d.app_start(WdaAppBundleId)
|
||||||
d.home()
|
d.home()
|
||||||
# time.sleep(2)
|
time.sleep(2)
|
||||||
# d.app_start(tikTokApp)
|
# d.app_start(tikTokApp)
|
||||||
target = self.relayDeviceScreenPort()
|
target = self.relayDeviceScreenPort()
|
||||||
self.pidList.append({
|
self.pidList.append({
|
||||||
|
|||||||
@@ -1,7 +1,11 @@
|
|||||||
from Module.DeviceInfo import Deviceinfo
|
from Module.DeviceInfo import Deviceinfo
|
||||||
from Module.FlaskSubprocessManager import FlaskSubprocessManager
|
from Module.FlaskSubprocessManager import FlaskSubprocessManager
|
||||||
|
from Utils.LogManager import LogManager
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|
||||||
|
LogManager.clearLogs()
|
||||||
|
|
||||||
print("启动flask")
|
print("启动flask")
|
||||||
manager = FlaskSubprocessManager.get_instance()
|
manager = FlaskSubprocessManager.get_instance()
|
||||||
manager.start()
|
manager.start()
|
||||||
|
|||||||
124
Utils/AiUtils.py
@@ -1,11 +1,53 @@
|
|||||||
|
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
|
import cv2
|
||||||
|
import numpy as np
|
||||||
|
|
||||||
|
|
||||||
# 工具类
|
# 工具类
|
||||||
class AiUtils(object):
|
class AiUtils(object):
|
||||||
|
|
||||||
|
# 在屏幕中找到对应的图片
|
||||||
|
@classmethod
|
||||||
|
def findImageInScreen(cls, target):
|
||||||
|
# 加载原始图像和模板图像
|
||||||
|
image_path = AiUtils.imagePathWithName("bgv") # 替换为你的图像路径
|
||||||
|
template_path = AiUtils.imagePathWithName(target) # 替换为你的模板路径
|
||||||
|
|
||||||
|
# 读取图像和模板,确保它们都是单通道灰度图
|
||||||
|
image = cv2.imread(image_path, cv2.IMREAD_GRAYSCALE)
|
||||||
|
template = cv2.imread(template_path, cv2.IMREAD_GRAYSCALE)
|
||||||
|
|
||||||
|
if image is None:
|
||||||
|
raise ValueError("背景图无法加载")
|
||||||
|
|
||||||
|
if template is None:
|
||||||
|
raise ValueError("模板图无法加载")
|
||||||
|
|
||||||
|
|
||||||
|
# 获取模板的宽度和高度
|
||||||
|
w, h = template.shape[::-1]
|
||||||
|
|
||||||
|
# 使用模板匹配方法
|
||||||
|
res = cv2.matchTemplate(image, template, cv2.TM_CCOEFF_NORMED)
|
||||||
|
threshold = 0.7 # 匹配度阈值,可以根据需要调整
|
||||||
|
loc = np.where(res >= threshold)
|
||||||
|
|
||||||
|
# 检查是否有匹配结果
|
||||||
|
if loc[0].size > 0:
|
||||||
|
# 取第一个匹配位置
|
||||||
|
pt = zip(*loc[::-1]).__next__() # 获取第一个匹配点的坐标
|
||||||
|
center_x = int(pt[0] + w // 2)
|
||||||
|
center_y = int(pt[1] + h // 2)
|
||||||
|
# print(f"第一个匹配到的小心心中心坐标: ({center_x}, {center_y})")
|
||||||
|
return center_x, center_y
|
||||||
|
else:
|
||||||
|
print("未找到匹配的目标")
|
||||||
|
return -1, -1
|
||||||
|
|
||||||
|
|
||||||
|
# 使用正则查找字符串中的数字
|
||||||
@classmethod
|
@classmethod
|
||||||
def findNumber(cls, str):
|
def findNumber(cls, str):
|
||||||
# 使用正则表达式匹配数字
|
# 使用正则表达式匹配数字
|
||||||
@@ -15,15 +57,91 @@ class AiUtils(object):
|
|||||||
return None # 如果没有找到数字,返回 None
|
return None # 如果没有找到数字,返回 None
|
||||||
|
|
||||||
|
|
||||||
# 根据名称获取图片地址
|
# 选择截图
|
||||||
@classmethod
|
@classmethod
|
||||||
def pathWithName(cls, name):
|
def screenshot(cls):
|
||||||
|
image_path = AiUtils.imagePathWithName("bgv") # 替换为你的图像路径
|
||||||
|
image = cv2.imread(image_path)
|
||||||
|
|
||||||
|
# 如果图像过大,缩小显示
|
||||||
|
scale_percent = 50 # 缩小比例
|
||||||
|
width = int(image.shape[1] * scale_percent / 100)
|
||||||
|
height = int(image.shape[0] * scale_percent / 100)
|
||||||
|
dim = (width, height)
|
||||||
|
resized_image = cv2.resize(image, dim, interpolation=cv2.INTER_AREA)
|
||||||
|
|
||||||
|
# 创建一个窗口并显示缩小后的图像
|
||||||
|
cv2.namedWindow("Image")
|
||||||
|
cv2.imshow("Image", resized_image)
|
||||||
|
print("请在图像上选择爱心图标区域,然后按Enter键确认。")
|
||||||
|
|
||||||
|
# 使用selectROI函数手动选择区域
|
||||||
|
roi = cv2.selectROI("Image", resized_image, showCrosshair=True, fromCenter=False)
|
||||||
|
|
||||||
|
# 将ROI坐标按原始图像尺寸放大
|
||||||
|
x, y, w, h = roi
|
||||||
|
x = int(x * image.shape[1] / resized_image.shape[1])
|
||||||
|
y = int(y * image.shape[0] / resized_image.shape[0])
|
||||||
|
w = int(w * image.shape[1] / resized_image.shape[1])
|
||||||
|
h = int(h * image.shape[0] / resized_image.shape[0])
|
||||||
|
|
||||||
|
# 根据选择的ROI提取爱心图标
|
||||||
|
if w > 0 and h > 0: # 确保选择的区域有宽度和高度
|
||||||
|
heart_icon = image[y:y + h, x:x + w]
|
||||||
|
|
||||||
|
# 转换为HSV颜色空间
|
||||||
|
hsv = cv2.cvtColor(heart_icon, cv2.COLOR_BGR2HSV)
|
||||||
|
|
||||||
|
# 定义红色的HSV范围
|
||||||
|
lower_red1 = np.array([0, 120, 70])
|
||||||
|
upper_red1 = np.array([10, 255, 255])
|
||||||
|
lower_red2 = np.array([170, 120, 70])
|
||||||
|
upper_red2 = np.array([180, 255, 255])
|
||||||
|
|
||||||
|
# 创建掩模
|
||||||
|
mask1 = cv2.inRange(hsv, lower_red1, upper_red1)
|
||||||
|
mask2 = cv2.inRange(hsv, lower_red2, upper_red2)
|
||||||
|
mask = mask1 + mask2
|
||||||
|
|
||||||
|
# 反转掩模,因为我们想要的是爱心图标,而不是背景
|
||||||
|
mask_inv = cv2.bitwise_not(mask)
|
||||||
|
|
||||||
|
# 应用掩模
|
||||||
|
heart_icon = cv2.bitwise_and(heart_icon, heart_icon, mask=mask_inv)
|
||||||
|
|
||||||
|
# 创建一个全透明的背景
|
||||||
|
height, width, channels = heart_icon.shape
|
||||||
|
roi = np.zeros((height, width, channels), dtype=np.uint8)
|
||||||
|
|
||||||
|
# 将爱心图标粘贴到透明背景上
|
||||||
|
for c in range(channels):
|
||||||
|
roi[:, :, c] = np.where(mask_inv == 255, heart_icon[:, :, c], roi[:, :, c])
|
||||||
|
|
||||||
|
# 图片名称
|
||||||
|
imgName = "temp.png"
|
||||||
|
|
||||||
|
# 保存结果
|
||||||
|
cv2.imwrite(imgName, roi)
|
||||||
|
|
||||||
|
# 显示结果
|
||||||
|
cv2.imshow("Heart Icon with Transparent Background", roi)
|
||||||
|
cv2.waitKey(0)
|
||||||
|
cv2.destroyAllWindows()
|
||||||
|
else:
|
||||||
|
print("未选择有效区域。")
|
||||||
|
|
||||||
|
|
||||||
|
# 根据名称获取图片完整地址
|
||||||
|
@classmethod
|
||||||
|
def imagePathWithName(cls, name):
|
||||||
current_file_path = os.path.abspath(__file__)
|
current_file_path = os.path.abspath(__file__)
|
||||||
# 获取当前文件所在的目录(即script目录)
|
# 获取当前文件所在的目录(即script目录)
|
||||||
current_dir = os.path.dirname(current_file_path)
|
current_dir = os.path.dirname(current_file_path)
|
||||||
# 由于script目录位于项目根目录下一级,因此需要向上一级目录移动两次
|
# 由于script目录位于项目根目录下一级,因此需要向上一级目录移动两次
|
||||||
project_root = os.path.abspath(os.path.join(current_dir, '..'))
|
project_root = os.path.abspath(os.path.join(current_dir, '..'))
|
||||||
# 构建资源文件的完整路径,向上两级目录,然后进入 resources 目录
|
# 构建资源文件的完整路径,向上两级目录,然后进入 resources 目录
|
||||||
resource_path = os.path.abspath(os.path.join(project_root, 'resources', name + ".jpg")).replace('/', '\\')
|
resource_path = os.path.abspath(os.path.join(project_root, 'resources', name + ".png")).replace('/', '\\')
|
||||||
return resource_path
|
return resource_path
|
||||||
|
|
||||||
|
|
||||||
|
# AiUtils.screenshot()
|
||||||
68
Utils/ControlUtils.py
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
from wda import Client
|
||||||
|
|
||||||
|
from Entity.Variables import tikTokApp
|
||||||
|
from Utils.AiUtils import AiUtils
|
||||||
|
from Utils.LogManager import LogManager
|
||||||
|
|
||||||
|
# 页面控制工具类
|
||||||
|
class ControlUtils(object):
|
||||||
|
|
||||||
|
# 打开Tik Tok
|
||||||
|
@classmethod
|
||||||
|
def openTikTok(cls, session: Client):
|
||||||
|
currentApp = session.app_current()
|
||||||
|
if currentApp != tikTokApp:
|
||||||
|
session.app_start(tikTokApp)
|
||||||
|
|
||||||
|
# 返回
|
||||||
|
@classmethod
|
||||||
|
def clickBack(cls, session: Client):
|
||||||
|
x, y = AiUtils.findImageInScreen("back")
|
||||||
|
if x > 0:
|
||||||
|
r = session.swipe_right()
|
||||||
|
if r.status == 0:
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
print("本页面无法返回")
|
||||||
|
LogManager.info("没有返回按钮")
|
||||||
|
return False
|
||||||
|
|
||||||
|
# 点赞
|
||||||
|
@classmethod
|
||||||
|
def clickLike(cls, session: Client):
|
||||||
|
scale = session.scale
|
||||||
|
x, y = AiUtils.findImageInScreen("add")
|
||||||
|
print(x,y)
|
||||||
|
if x > -1:
|
||||||
|
print("找到目标了")
|
||||||
|
r = session.click(x // scale, y // scale + 50)
|
||||||
|
if r.status == 0:
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
print("没有找到目标")
|
||||||
|
return False
|
||||||
|
|
||||||
|
# 点击评论
|
||||||
|
# @classmethod
|
||||||
|
# def clickComment(cls, session: Client):
|
||||||
|
# scale = session.scale
|
||||||
|
# x, y = AiUtils.findImageInScreen("add")
|
||||||
|
# print(x,y)
|
||||||
|
# if x > -1:
|
||||||
|
# print("找到目标了")
|
||||||
|
# r = session.click(x // scale, y // scale + 120)
|
||||||
|
# if r.status == 0:
|
||||||
|
# return True
|
||||||
|
# else:
|
||||||
|
# return False
|
||||||
|
# else:
|
||||||
|
# print("没有找到目标")
|
||||||
|
# return False
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# ControlUtils.backAction("eca000fcb6f55d7ed9b4c524055214c26a7de7aa")
|
||||||
27
Utils/ThreadManager.py
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
from threading import Thread, Event
|
||||||
|
|
||||||
|
from Utils.LogManager import LogManager
|
||||||
|
from script.ScriptManager import ScriptManager
|
||||||
|
|
||||||
|
|
||||||
|
class ThreadManager():
|
||||||
|
threads = {}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def add(cls, udid, t: Thread, stopEvent: Event):
|
||||||
|
if udid in cls.threads:
|
||||||
|
print("▲ 线程已存在")
|
||||||
|
return
|
||||||
|
cls.threads[udid] = {"thread": t, "stopEvent": stopEvent}
|
||||||
|
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def stop(cls, udid):
|
||||||
|
info = cls.threads[udid]
|
||||||
|
if info:
|
||||||
|
info["stopEvent"].set() # 停止线程
|
||||||
|
info["thread"].join(timeout=3) # 等待线程退出
|
||||||
|
del cls.threads[udid]
|
||||||
|
LogManager.info("停止线程成功")
|
||||||
|
else:
|
||||||
|
LogManager.info("无此线程,无需关闭")
|
||||||
|
Before Width: | Height: | Size: 1.8 KiB |
BIN
resources/add.png
Normal file
|
After Width: | Height: | Size: 899 B |
BIN
resources/advertisement.png
Normal file
|
After Width: | Height: | Size: 1.8 KiB |
BIN
resources/back.png
Normal file
|
After Width: | Height: | Size: 1.1 KiB |
|
Before Width: | Height: | Size: 70 KiB |
BIN
resources/bgv.png
Normal file
|
After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1007 B |
BIN
resources/comment.png
Normal file
|
After Width: | Height: | Size: 5.6 KiB |
|
Before Width: | Height: | Size: 903 B |
BIN
resources/like.png
Normal file
|
After Width: | Height: | Size: 5.1 KiB |
|
Before Width: | Height: | Size: 1.3 KiB |
|
Before Width: | Height: | Size: 1023 B |
BIN
resources/search.png
Normal file
|
After Width: | Height: | Size: 3.9 KiB |
@@ -1,7 +1,9 @@
|
|||||||
|
import random
|
||||||
|
import time
|
||||||
import wda
|
import wda
|
||||||
|
|
||||||
from Utils.AiUtils import AiUtils
|
from Utils.AiUtils import AiUtils
|
||||||
|
from Utils.ControlUtils import ControlUtils
|
||||||
|
|
||||||
|
|
||||||
# 脚本管理类
|
# 脚本管理类
|
||||||
@@ -19,23 +21,52 @@ class ScriptManager():
|
|||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
super().__init__()
|
super().__init__()
|
||||||
# 脚本开关
|
|
||||||
self.running = False
|
|
||||||
self.initialized = True # 标记已初始化
|
self.initialized = True # 标记已初始化
|
||||||
|
|
||||||
# 养号
|
# 养号
|
||||||
def growAccount(self, udid):
|
def growAccount(self, udid, event):
|
||||||
client = wda.USBClient(udid)
|
client = wda.USBClient(udid)
|
||||||
session = client.session()
|
session = client.session()
|
||||||
session.appium_settings({"snapshotMaxDepth": 15})
|
session.appium_settings({"snapshotMaxDepth": 15})
|
||||||
numberLabel = session.xpath("//*[@name='a11y_vo_inbox']")
|
|
||||||
if numberLabel:
|
# 检测当前打开的是否是Tik Tok。如果不是就打开
|
||||||
content = numberLabel.label
|
ControlUtils.openTikTok(session)
|
||||||
number = AiUtils.findNumber(content)
|
time.sleep(3)
|
||||||
print(number)
|
|
||||||
|
# 假设此时有个开关
|
||||||
|
while not event.is_set():
|
||||||
|
try:
|
||||||
|
img = client.screenshot()
|
||||||
|
filePath = "resources/bgv.png"
|
||||||
|
img.save(filePath)
|
||||||
|
except Exception as e:
|
||||||
|
print(e)
|
||||||
|
|
||||||
|
# 判断视频类型。普通视频、广告、直播,只有普通视频停留。其他视频全部划走
|
||||||
|
addX, addY = AiUtils.findImageInScreen("add")
|
||||||
|
# 如果找到普通视频
|
||||||
|
if addX > 0:
|
||||||
|
needLike = random.randint(0, 10)
|
||||||
|
|
||||||
|
# 随机时间间隔
|
||||||
|
videoTime = random.randint(2,5)
|
||||||
|
time.sleep(videoTime)
|
||||||
|
|
||||||
|
# 百分之三的概率点赞
|
||||||
|
if needLike < 3:
|
||||||
|
ControlUtils.clickLike(session)
|
||||||
|
|
||||||
|
# 随机时间间隔
|
||||||
|
# videoTime = random.randint(10,30)
|
||||||
|
# time.sleep(videoTime)
|
||||||
|
session.swipe_up()
|
||||||
else:
|
else:
|
||||||
print("没找到")
|
# 随机时间
|
||||||
|
nextTime = random.randint(1,5)
|
||||||
|
time.sleep(nextTime)
|
||||||
|
session.swipe_up()
|
||||||
|
continue
|
||||||
|
|
||||||
manager = ScriptManager()
|
# manager = ScriptManager()
|
||||||
manager.growAccount("eca000fcb6f55d7ed9b4c524055214c26a7de7aa")
|
# manager.growAccount("eca000fcb6f55d7ed9b4c524055214c26a7de7aa")
|
||||||
|
|
||||||
|
|||||||