You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
441 lines
17 KiB
441 lines
17 KiB
3 months ago
|
<?php
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | Copyright (c) 2016~2020 https://www.crmeb.com All rights reserved.
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | Author: CRMEB Team <admin@crmeb.com>
|
||
|
// +----------------------------------------------------------------------
|
||
|
|
||
|
namespace app\services\work;
|
||
|
|
||
|
|
||
|
use app\dao\work\WorkGroupChatDao;
|
||
|
use app\jobs\work\WorkClientJob;
|
||
|
use app\jobs\work\WorkGroupChatJob;
|
||
|
use app\services\BaseServices;
|
||
|
use crmeb\services\wechat\config\WorkConfig;
|
||
|
use crmeb\services\wechat\Work;
|
||
|
use crmeb\traits\ServicesTrait;
|
||
|
use think\db\exception\DataNotFoundException;
|
||
|
use think\db\exception\DbException;
|
||
|
use think\db\exception\ModelNotFoundException;
|
||
|
use think\exception\ValidateException;
|
||
|
|
||
|
/**
|
||
|
* Class WorkGroupChatServices
|
||
|
* @package app\services\work
|
||
|
* @mixin WorkGroupChatDao
|
||
|
*/
|
||
|
class WorkGroupChatServices extends BaseServices
|
||
|
{
|
||
|
|
||
|
use ServicesTrait;
|
||
|
|
||
|
/**
|
||
|
* WorkGroupChatServices constructor.
|
||
|
* @param WorkGroupChatDao $dao
|
||
|
*/
|
||
|
public function __construct(WorkGroupChatDao $dao)
|
||
|
{
|
||
|
$this->dao = $dao;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取客户群群发列表
|
||
|
* @param array $where
|
||
|
* @param array $with
|
||
|
* @return array
|
||
|
* @throws DataNotFoundException
|
||
|
* @throws DbException
|
||
|
* @throws ModelNotFoundException
|
||
|
*/
|
||
|
public function groupChatList(array $where, array $with = [])
|
||
|
{
|
||
|
[$page, $limit] = $this->getPageValue();
|
||
|
$list = $this->dao->groupChat($where)->with($with)->page($page, $limit)->select()->toArray();
|
||
|
$count = $this->dao->groupChat($where)->count();
|
||
|
return compact('list', 'count');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 群发详情列表
|
||
|
* @param array $chatIds
|
||
|
* @return array
|
||
|
* @throws DataNotFoundException
|
||
|
* @throws DbException
|
||
|
* @throws ModelNotFoundException
|
||
|
*/
|
||
|
public function getOwnerChatList(array $chatIds, $status = null)
|
||
|
{
|
||
|
return $this->groupChatList(['chat_id' => $chatIds, 'status' => $status], [
|
||
|
'sendResult' => function ($query) {
|
||
|
$query->field(['chat_id', 'status']);
|
||
|
}, 'chatMember' => function ($query) {
|
||
|
$query->field(['count(*) as sun', 'group_id']);
|
||
|
}
|
||
|
]);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 同步企业微信客户群
|
||
|
* @param string|null $nextCursor
|
||
|
* @return bool
|
||
|
* @author 等风来
|
||
|
* @email 136327134@qq.com
|
||
|
* @date 2022/10/10
|
||
|
*/
|
||
|
public function authGroupChat(string $nextCursor = null)
|
||
|
{
|
||
|
/** @var WorkConfig $confg */
|
||
|
$config = app()->make(WorkConfig::class);
|
||
|
$corpId = $config->get('corpId');
|
||
|
if (!$corpId) {
|
||
|
throw new ValidateException('请先配置企业微信ID');
|
||
|
}
|
||
|
$res = Work::getGroupChats([], 100, $nextCursor);
|
||
|
if (0 !== ($res['errcode'] ?? 0)) {
|
||
|
throw new ValidateException($res['errmsg']);
|
||
|
}
|
||
|
$groupChatList = $res['group_chat_list'] ?? [];
|
||
|
if ($groupChatList) {
|
||
|
$groupChat = [];
|
||
|
foreach ($groupChatList as $item) {
|
||
|
$item['corp_id'] = $corpId;
|
||
|
if (($id = $this->dao->value(['chat_id' => $item['chat_id'], 'corp_id' => $corpId], 'id'))) {
|
||
|
$this->dao->update($id, $item);
|
||
|
} else {
|
||
|
$item['create_time'] = time();
|
||
|
$groupChat[] = $item;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($groupChat) {
|
||
|
$this->dao->saveAll($groupChat);
|
||
|
}
|
||
|
|
||
|
foreach ($groupChatList as $item) {
|
||
|
WorkGroupChatJob::dispatchDo('authChat', [$corpId, $item['chat_id']]);
|
||
|
}
|
||
|
|
||
|
//如果有下一页继续执行
|
||
|
if (!empty($res['next_cursor'])) {
|
||
|
WorkGroupChatJob::dispatchDo('authGroupChat', [$res['next_cursor']]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 保存群详情
|
||
|
* @param string $corpId
|
||
|
* @param string $chatId
|
||
|
* @return bool
|
||
|
* @throws DataNotFoundException
|
||
|
* @throws DbException
|
||
|
* @throws ModelNotFoundException
|
||
|
*/
|
||
|
public function saveWorkGroupChat(string $corpId, string $chatId)
|
||
|
{
|
||
|
$response = Work::getGroupChat($chatId);
|
||
|
if (0 !== $response['errcode']) {
|
||
|
throw new ValidateException($response['errmsg']);
|
||
|
}
|
||
|
$groupInfo = $response['group_chat'] ?? [];
|
||
|
$groupInfo['admin_list'] = json_encode(array_column($groupInfo['admin_list'], 'userid'));
|
||
|
$memberList = $groupInfo['member_list'] ?? [];
|
||
|
unset($groupInfo['member_list']);
|
||
|
$group = $this->dao->get(['corp_id' => $corpId, 'chat_id' => $chatId]);
|
||
|
return $this->transaction(function () use ($chatId, $corpId, $group, $groupInfo, $memberList) {
|
||
|
if ($group) {
|
||
|
$group->name = $groupInfo['name'];
|
||
|
$group->owner = $groupInfo['owner'];
|
||
|
$group->notice = $groupInfo['notice'] ?? '';
|
||
|
$group->group_create_time = $groupInfo['create_time'];
|
||
|
$group->member_num = count($memberList);
|
||
|
$group->save();
|
||
|
} else {
|
||
|
$group = $this->dao->save([
|
||
|
'corp_id' => $corpId,
|
||
|
'chat_id' => $chatId,
|
||
|
'name' => $groupInfo['name'],
|
||
|
'owner' => $groupInfo['owner'],
|
||
|
'notice' => $groupInfo['notice'] ?? '',
|
||
|
'member_num' => count($memberList),
|
||
|
'group_create_time' => $groupInfo['create_time'],
|
||
|
'status' => $groupInfo['status'] ?? 0,
|
||
|
]);
|
||
|
}
|
||
|
$this->saveMember($memberList, $group->id, $group->member_num);
|
||
|
return $group->id;
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param array $where
|
||
|
* @return array
|
||
|
*/
|
||
|
public function getList(array $where)
|
||
|
{
|
||
|
[$page, $limit] = $this->getPageValue();
|
||
|
$where['timeKey'] = 'group_create_time';
|
||
|
$where['status'] = [0, 2, 3];
|
||
|
$list = $this->dao->getDataList($where, ['*'], $page, $limit, 'group_create_time', [
|
||
|
'ownerInfo' => function ($query) {
|
||
|
$query->field(['userid', 'name']);
|
||
|
},
|
||
|
]);
|
||
|
|
||
|
//提取管理员数据
|
||
|
$adminUserId = [];
|
||
|
foreach ($list as $item) {
|
||
|
$adminUserId = array_merge($adminUserId, $item['admin_list'] ?? []);
|
||
|
}
|
||
|
$adminUserId = array_merge(array_unique(array_filter($adminUserId)));
|
||
|
if ($adminUserId) {
|
||
|
/** @var WorkMemberServices $memberService */
|
||
|
$memberService = app()->make(WorkMemberServices::class);
|
||
|
$adminUserList = $memberService->getColumn([
|
||
|
['userid', 'in', $adminUserId],
|
||
|
], 'name', 'userid');
|
||
|
foreach ($list as &$item) {
|
||
|
$newAdminUser = [];
|
||
|
if (!empty($item['admin_list'])) {
|
||
|
foreach ($adminUserList as $key => $value) {
|
||
|
if (in_array($key, $item['admin_list'])) {
|
||
|
$newAdminUser[] = ['name' => $value, 'userid' => $key];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
$item['admin_user_list'] = $newAdminUser;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$count = $this->dao->count();
|
||
|
return compact('list', 'count');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 企业微信客户群变动
|
||
|
* @param array $payload
|
||
|
* @throws DataNotFoundException
|
||
|
* @throws DbException
|
||
|
* @throws ModelNotFoundException
|
||
|
*/
|
||
|
public function updateGroupChat(array $payload)
|
||
|
{
|
||
|
$corpId = $payload['ToUserName'];
|
||
|
$chatId = $payload['ChatId'];
|
||
|
|
||
|
$groupInfo = $this->dao->get(['corp_id' => $corpId, 'chat_id' => $chatId]);
|
||
|
if (!$groupInfo) {
|
||
|
$groupId = $this->saveWorkGroupChat($corpId, $chatId);
|
||
|
$groupInfo = $this->dao->get($groupId);
|
||
|
}
|
||
|
$response = Work::getGroupChat($chatId);
|
||
|
if (0 !== $response['errcode']) {
|
||
|
throw new ValidateException($response['errmsg'] ?? '企业微信查询群详情失败');
|
||
|
}
|
||
|
$groupChat = $response['group_chat'] ?? [];
|
||
|
$memberList = $groupChat['member_list'];
|
||
|
|
||
|
/** @var WorkGroupChatStatisticServices $statisticService */
|
||
|
$statisticService = app()->make(WorkGroupChatStatisticServices::class);
|
||
|
|
||
|
$this->transaction(function () use ($payload, $groupInfo, $groupChat, $memberList, $statisticService) {
|
||
|
switch ($payload['UpdateDetail']) {
|
||
|
case 'add_member':
|
||
|
$groupInfo->member_num++;
|
||
|
$this->saveMember($memberList, $groupInfo->id, $groupInfo->member_num, true);
|
||
|
$statisticService->saveOrUpdate($groupInfo->id, true, false, $groupInfo->member_num, $groupInfo->retreat_group_num);
|
||
|
break;
|
||
|
case 'del_member':
|
||
|
$groupInfo->member_num--;
|
||
|
$groupInfo->retreat_group_num++;
|
||
|
$this->saveMember($memberList, $groupInfo->id, $groupInfo->member_num, false);
|
||
|
$statisticService->saveOrUpdate($groupInfo->id, false, true, $groupInfo->member_num, $groupInfo->retreat_group_num);
|
||
|
break;
|
||
|
case 'change_owner':
|
||
|
$groupInfo->owner = $groupChat['owner'];
|
||
|
break;
|
||
|
case 'change_name':
|
||
|
$groupInfo->name = $groupChat['name'];
|
||
|
break;
|
||
|
case 'change_notice':
|
||
|
$groupInfo->notice = $groupChat['notice'];
|
||
|
break;
|
||
|
}
|
||
|
if (!empty($groupChat['admin_list'])) {
|
||
|
$groupInfo->admin_list = json_encode(array_column($groupChat['admin_list'], 'userid'));
|
||
|
}
|
||
|
$groupInfo->save();
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 保存群成员
|
||
|
* @param array $memberList
|
||
|
* @param int $groupId
|
||
|
* @param int $sum
|
||
|
* @param bool $plus
|
||
|
* @return mixed
|
||
|
*/
|
||
|
public function saveMember(array $memberList, int $groupId, int $sum = 0, bool $plus = false)
|
||
|
{
|
||
|
$data = [];
|
||
|
/** @var WorkGroupChatMemberServices $chatMemberService */
|
||
|
$chatMemberService = app()->make(WorkGroupChatMemberServices::class);
|
||
|
$newUserIds = array_column($memberList, 'userid');
|
||
|
$userids = $chatMemberService->getColumn(['group_id' => $groupId], 'userid');
|
||
|
$unUserIds = array_diff($userids, $newUserIds);
|
||
|
$labelList = [];
|
||
|
foreach ($memberList as $item) {
|
||
|
$item['group_id'] = $groupId;
|
||
|
$state = $item['state'] ?? '';
|
||
|
if (isset($item['state'])) {
|
||
|
unset($item['state']);
|
||
|
}
|
||
|
$item['invitor_userid'] = $item['invitor']['userid'] ?? '';
|
||
|
$unionid = $item['unionid'] ?? '';
|
||
|
unset($item['invitor'], $item['unionid']);
|
||
|
if ($chatMemberService->count(['group_id' => $groupId, 'userid' => $item['userid']])) {
|
||
|
$chatMemberService->update(['group_id' => $groupId, 'userid' => $item['userid']], [
|
||
|
'type' => $item['type'],
|
||
|
'unionid' => $unionid,
|
||
|
'chat_sum' => $sum,
|
||
|
'status' => 1,
|
||
|
'join_time' => $item['join_time'],
|
||
|
'join_scene' => $item['join_scene'],
|
||
|
'invitor_userid' => $item['invitor_userid'],
|
||
|
'group_nickname' => $item['group_nickname'],
|
||
|
]);
|
||
|
} else {
|
||
|
if ($state) {
|
||
|
$labelList[] = ['userid' => $item['userid'], 'type' => $item['type'], 'state' => $state];
|
||
|
}
|
||
|
$item['unionid'] = $unionid;
|
||
|
$item['chat_sum'] = $sum;
|
||
|
$item['state'] = $state;
|
||
|
$item['create_time'] = time();
|
||
|
$data[] = $item;
|
||
|
}
|
||
|
}
|
||
|
if ($data) {
|
||
|
$chatMemberService->saveAll($data);
|
||
|
//如果没有客户信息同步客户信息
|
||
|
/** @var WorkClientServices $clientService */
|
||
|
$clientService = app()->make(WorkClientServices::class);
|
||
|
$corpId = app()->make(WorkConfig::class)->get('corpId');
|
||
|
foreach ($data as $item) {
|
||
|
if (2 == $item['type'] && $clientService->count(['external_userid' => $item['userid']])) {
|
||
|
WorkClientJob::dispatchDo('saveClientInfo', [$corpId, $item['userid'], '']);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ($unUserIds) {
|
||
|
$chatMemberService->update([
|
||
|
['userid', 'in', $unUserIds]
|
||
|
], ['status' => 0]);
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 解散客户群
|
||
|
* @param string $corpId
|
||
|
* @param string $chatId
|
||
|
* @return mixed
|
||
|
* @throws DataNotFoundException
|
||
|
* @throws DbException
|
||
|
* @throws ModelNotFoundException
|
||
|
*/
|
||
|
public function dismissGroupChat(string $corpId, string $chatId)
|
||
|
{
|
||
|
$groupChat = $this->dao->get(['corp_id' => $corpId, 'chat_id' => $chatId], ['id']);
|
||
|
if (!$groupChat) {
|
||
|
throw new ValidateException('没有查询到群');
|
||
|
}
|
||
|
return $this->transaction(function () use ($groupChat) {
|
||
|
/** @var WorkGroupChatMemberServices $chatMemberService */
|
||
|
$chatMemberService = app()->make(WorkGroupChatMemberServices::class);
|
||
|
$chatMemberService->delete(['group_id' => $groupChat->id]);
|
||
|
return $groupChat->delete();
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 群统计
|
||
|
* @param int $id
|
||
|
* @param string $time
|
||
|
* @return array
|
||
|
*/
|
||
|
public function getChatStatistics(int $id, string $time)
|
||
|
{
|
||
|
/** @var WorkGroupChatMemberServices $chatMemberService */
|
||
|
$chatMemberService = app()->make(WorkGroupChatMemberServices::class);
|
||
|
$data = [
|
||
|
'toDaySum' => $chatMemberService->getToDaySum($id),
|
||
|
'toDayReturn' => $chatMemberService->getToDayReturn($id),
|
||
|
'groupChatSum' => $this->dao->value(['id' => $id], 'member_num'),
|
||
|
'groupChatReturnSum' => $this->dao->value(['id' => $id], 'retreat_group_num')
|
||
|
];
|
||
|
$data['groupChatList'] = $chatMemberService->getChatMemberStatistics($id, 'join_time', ['count(*) as sum'], 1, $time);
|
||
|
$data['groupChatReturnList'] = $chatMemberService->getChatMemberStatistics($id, 'join_time', ['count(*) as sum'], 0, $time);
|
||
|
return $data;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 群成员统计
|
||
|
* @param int $id
|
||
|
* @param string $time
|
||
|
* @return array
|
||
|
*/
|
||
|
public function getChatStatisticsList(int $id, string $time)
|
||
|
{
|
||
|
[$page, $limit] = $this->getPageValue();
|
||
|
/** @var WorkGroupChatMemberServices $chatMemberService */
|
||
|
$chatMemberService = app()->make(WorkGroupChatMemberServices::class);
|
||
|
$newCount = $chatMemberService->getChatMemberStatisticsCount($id, 1, $time);
|
||
|
$returnCount = $chatMemberService->getChatMemberStatisticsCount($id, 0, $time);
|
||
|
$SumCount = $newCount > $returnCount ? $newCount : $returnCount;
|
||
|
$groupChatList = $chatMemberService->getChatMemberStatistics($id, 'join_time', ['count(*) as sum', 'chat_sum', 'retreat_chat_num'], 1, $time, $page, $limit);
|
||
|
$groupChatReturnList = $chatMemberService->getChatMemberStatistics($id, 'join_time', ['count(*) as retreat_sum'], 0, $time, $page, $limit);
|
||
|
$count = ($rCount = count($groupChatReturnList)) > ($count = count($groupChatList)) ? $rCount : $count;
|
||
|
$data = [];
|
||
|
for ($i = 0; $i < $count; $i++) {
|
||
|
$data[] = [
|
||
|
'time' => $groupChatList[$i]['time'] ?? $groupChatReturnList[$i]['time'] ?? '',
|
||
|
'sum' => $groupChatList[$i]['sum'] ?? 0,
|
||
|
'retreat_chat_num' => $groupChatList[$i]['retreat_chat_num'] ?? 0,
|
||
|
'chat_sum' => $groupChatList[$i]['chat_sum'] ?? 0,
|
||
|
'retreat_sum' => $groupChatReturnList[$i]['retreat_sum'] ?? 0,
|
||
|
];
|
||
|
}
|
||
|
return ['list' => $data, 'count' => $SumCount];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $chatId
|
||
|
* @param string $corpId
|
||
|
* @return array
|
||
|
* @throws DataNotFoundException
|
||
|
* @throws DbException
|
||
|
* @throws ModelNotFoundException
|
||
|
*/
|
||
|
public function getGroupInfo(string $chatId, string $corpId)
|
||
|
{
|
||
|
$groupInfo = $this->dao->get(['chat_id' => $chatId, 'corp_id' => $corpId]);
|
||
|
if (!$groupInfo) {
|
||
|
throw new ValidateException('客户群未查到');
|
||
|
}
|
||
|
/** @var WorkGroupChatMemberServices $service */
|
||
|
$service = app()->make(WorkGroupChatMemberServices::class);
|
||
|
$groupInfo['todaySum'] = $service->getToDaySum($groupInfo->id);
|
||
|
$groupInfo['todayReturnSum'] = $service->getToDayReturn($groupInfo->id);
|
||
|
return $groupInfo->toArray();
|
||
|
}
|
||
|
}
|