2025-07-08 09:52:59 +08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
namespace app\command;
|
|
|
|
|
|
|
|
|
|
use app\job\WorkbenchAutoLikeJob;
|
|
|
|
|
use think\console\Command;
|
|
|
|
|
use think\console\Input;
|
|
|
|
|
use think\console\input\Option;
|
|
|
|
|
use think\console\Output;
|
|
|
|
|
use think\facade\Cache;
|
|
|
|
|
use think\facade\Log;
|
|
|
|
|
use think\Queue;
|
|
|
|
|
use app\api\controller\AutomaticAssign;
|
|
|
|
|
|
|
|
|
|
class SwitchFriendsCommand extends Command
|
|
|
|
|
{
|
|
|
|
|
// 队列名称
|
|
|
|
|
protected $queueName = 'switch_friends';
|
|
|
|
|
|
|
|
|
|
protected function configure()
|
|
|
|
|
{
|
|
|
|
|
$this->setName('switch:friends')
|
|
|
|
|
->setDescription('切换好友命令');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected function execute(Input $input, Output $output)
|
|
|
|
|
{
|
|
|
|
|
$cacheKey = 'allotWechatFriend';
|
|
|
|
|
$now = time();
|
|
|
|
|
$maxRetry = 5;
|
|
|
|
|
$retry = 0;
|
|
|
|
|
$switchedIds = [];
|
2025-07-11 16:34:50 +08:00
|
|
|
$totalProcessed = 0;
|
|
|
|
|
$totalSuccess = 0;
|
|
|
|
|
$totalFailed = 0;
|
2025-07-08 09:52:59 +08:00
|
|
|
|
2025-07-11 16:34:50 +08:00
|
|
|
$output->writeln('开始执行好友切换任务...');
|
2025-07-08 09:52:59 +08:00
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
$friends = Cache::get($cacheKey, []);
|
|
|
|
|
|
|
|
|
|
$toSwitch = [];
|
|
|
|
|
foreach ($friends as $friend) {
|
2025-07-08 09:59:21 +08:00
|
|
|
if (isset($friend['time']) && $friend['time'] < $now) {
|
2025-07-08 09:52:59 +08:00
|
|
|
$toSwitch[] = $friend;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (empty($toSwitch)) {
|
|
|
|
|
$output->writeln('没有需要切换的好友');
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-11 16:34:50 +08:00
|
|
|
$output->writeln('找到 ' . count($toSwitch) . ' 个需要切换的好友');
|
|
|
|
|
|
2025-07-08 09:52:59 +08:00
|
|
|
$automaticAssign = new AutomaticAssign();
|
2025-07-11 16:34:50 +08:00
|
|
|
|
|
|
|
|
// 根据accountId对数组进行归类
|
|
|
|
|
$groupedByAccount = [];
|
2025-07-08 09:52:59 +08:00
|
|
|
foreach ($toSwitch as $friend) {
|
2025-07-11 16:34:50 +08:00
|
|
|
$accountId = $friend['accountId'];
|
|
|
|
|
if (!isset($groupedByAccount[$accountId])) {
|
|
|
|
|
$groupedByAccount[$accountId] = [];
|
|
|
|
|
}
|
2025-07-08 10:32:37 +08:00
|
|
|
$friendId = !empty($friend['friendId']) ? $friend['friendId'] : $friend['id'];
|
2025-07-11 16:34:50 +08:00
|
|
|
$groupedByAccount[$accountId][] = $friendId;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 对每个账号的好友进行20个为一组的分组
|
|
|
|
|
foreach ($groupedByAccount as $accountId => $accountFriends) {
|
|
|
|
|
$chunks = array_chunk($accountFriends, 20);
|
|
|
|
|
$output->writeln('账号 ' . $accountId . ' 共有 ' . count($accountFriends) . ' 个好友,分为 ' . count($chunks) . ' 组');
|
|
|
|
|
|
|
|
|
|
$accountSuccess = 0;
|
|
|
|
|
$accountFailed = 0;
|
|
|
|
|
|
|
|
|
|
foreach ($chunks as $chunkIndex => $chunk) {
|
|
|
|
|
$output->writeln('处理账号 ' . $accountId . ' 第 ' . ($chunkIndex + 1) . ' 组,共 ' . count($chunk) . ' 个好友');
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
$friendIds = implode(',', $chunk);
|
|
|
|
|
$res = $automaticAssign->multiAllotFriendToAccount([
|
|
|
|
|
'wechatFriendIds' => $friendIds,
|
|
|
|
|
'toAccountId' => $accountId,
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
$res = json_decode($res, true);
|
|
|
|
|
if ($res['code'] == 200){
|
|
|
|
|
$output->writeln('✓ 成功切换好友:' . $friendIds . ' 到账号:' . $accountId);
|
|
|
|
|
$switchedIds = array_merge($switchedIds, $chunk);
|
|
|
|
|
$accountSuccess += count($chunk);
|
|
|
|
|
$totalSuccess += count($chunk);
|
|
|
|
|
} else {
|
|
|
|
|
$output->writeln('✗ 切换失败 - 好友:' . $friendIds . ' 到账号:' . $accountId . ' 结果:' . $res['msg']);
|
|
|
|
|
$accountFailed += count($chunk);
|
|
|
|
|
$totalFailed += count($chunk);
|
|
|
|
|
}
|
|
|
|
|
} catch (\Exception $e) {
|
|
|
|
|
$output->writeln('✗ 切换异常 - 好友:' . implode(',', $chunk) . ' 到账号:' . $accountId . ' 错误:' . $e->getMessage());
|
|
|
|
|
Log::error('切换好友异常: ' . $e->getMessage() . ' 好友IDs: ' . implode(',', $chunk) . ' 账号ID: ' . $accountId);
|
|
|
|
|
$accountFailed += count($chunk);
|
|
|
|
|
$totalFailed += count($chunk);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$totalProcessed += count($chunk);
|
|
|
|
|
|
|
|
|
|
// 每组处理完后稍作延迟,避免请求过于频繁
|
|
|
|
|
if ($chunkIndex < count($chunks) - 1) {
|
|
|
|
|
sleep(1);
|
|
|
|
|
}
|
2025-07-08 11:56:24 +08:00
|
|
|
}
|
2025-07-11 16:34:50 +08:00
|
|
|
|
|
|
|
|
$output->writeln('账号 ' . $accountId . ' 处理完成 - 成功:' . $accountSuccess . ',失败:' . $accountFailed);
|
2025-07-08 09:52:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 过滤掉已切换的,保留未切换和新进来的
|
|
|
|
|
$newFriends = Cache::get($cacheKey, []);
|
|
|
|
|
$updated = [];
|
|
|
|
|
foreach ($newFriends as $friend) {
|
2025-07-08 10:32:37 +08:00
|
|
|
$friendId = !empty($friend['friendId']) ? $friend['friendId'] : $friend['id'];
|
|
|
|
|
if (!in_array($friendId, $switchedIds)) {
|
2025-07-08 09:52:59 +08:00
|
|
|
$updated[] = $friend;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 按time升序排序
|
|
|
|
|
usort($updated, function($a, $b) {
|
|
|
|
|
return ($a['time'] ?? 0) <=> ($b['time'] ?? 0);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$success = Cache::set($cacheKey, $updated);
|
|
|
|
|
$retry++;
|
|
|
|
|
} while (!$success && $retry < $maxRetry);
|
|
|
|
|
|
2025-07-11 16:34:50 +08:00
|
|
|
$output->writeln('=== 切换任务完成 ===');
|
|
|
|
|
$output->writeln('总处理数量:' . $totalProcessed);
|
|
|
|
|
$output->writeln('成功切换:' . $totalSuccess);
|
|
|
|
|
$output->writeln('切换失败:' . $totalFailed);
|
|
|
|
|
$output->writeln('成功率:' . ($totalProcessed > 0 ? round(($totalSuccess / $totalProcessed) * 100, 2) : 0) . '%');
|
|
|
|
|
$output->writeln('缓存已更新并排序');
|
2025-07-08 09:52:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|