Message.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  1. <?php
  2. /**
  3. * raingad IM [ThinkPHP6]
  4. * @author xiekunyu <raingad@foxmail.com>
  5. */
  6. namespace app\enterprise\model;
  7. use app\admin\model\KeywordLanguages;
  8. use app\admin\model\QuestionLanguages;
  9. use app\manage\model\Config;
  10. use app\BaseModel;
  11. use think\facade\Db;
  12. use think\facade\Cache;
  13. class Message extends BaseModel
  14. {
  15. protected $pk="msg_id";
  16. protected $json = ["extends"];
  17. protected $jsonAssoc = true;
  18. protected static $fileType=['file','image','video','voice','emoji'];
  19. //给用户发送自动消息
  20. public static function sendAutoReply($field, $user_id, $cs_uid, $language_code){
  21. //获取自动回复的消息内容
  22. $content = Config::getFieldValue($field, $language_code);
  23. // 如果设置了自动回复消息,则发送
  24. if($content){
  25. $userInfo=User::field('user_id,realname,avatar')->where(['user_id'=>$cs_uid])->find();//客服信息
  26. if($userInfo){
  27. $userInfo = $userInfo->toArray();
  28. $userInfo['dispalayName']=$userInfo['realname'];
  29. $userInfo['id']=$userInfo['user_id'];
  30. $userInfo['avatar']=avatarUrl($userInfo['avatar'],$userInfo['realname'],$userInfo['user_id']);
  31. $msg=[
  32. 'id'=>\utils\Str::getUuid(),
  33. 'user_id'=>$cs_uid,
  34. 'content'=>$content,
  35. 'toContactId'=>$user_id,
  36. 'sendTime'=>time()*1000,
  37. 'type'=>'event',
  38. 'is_group'=>0,
  39. 'status'=>'succeed',
  40. 'fromUser'=>$userInfo,
  41. 'at'=>[],
  42. ];
  43. self::sendMsg($msg,0,1);
  44. }
  45. }
  46. }
  47. // 添加聊天记录
  48. public static function addData($data){
  49. return Db::name('message')->insert($data);
  50. }
  51. // 更新消息状态
  52. public static function editData($update,$map){
  53. return Db::name('message')->where($map)->update($update);
  54. }
  55. // 查询聊天记录
  56. public static function getList($map,$where,$sort,$listRows,$pageSize){
  57. $list= Db::name('message')
  58. ->where($map)
  59. ->where($where)
  60. ->order($sort)
  61. ->paginate(['list_rows'=>$listRows,'page'=>$pageSize]);
  62. return $list;
  63. }
  64. // 发送消息
  65. public function sendMessage($param,$globalConfig=false){
  66. $param['language_code'] = $param['language_code'] ?? 'en';
  67. $is_group = $param['is_group'] ?? 0;
  68. $uid=self::$uid ? : ($param['user_id'] ?? 1);
  69. $is_robot = false; //是否给机器人发送消息
  70. if($param['toContactId']==-1){
  71. $is_group=0;
  72. }
  73. // 如果是系统账号,直接禁言
  74. if($is_group>1){
  75. $this->error=lang('im.forbidChat');
  76. return false;
  77. }
  78. $isForward=$param['is_forward'] ?? 0;
  79. $sendInterval = $globalConfig['chatInfo']['sendInterval'] ?? 0;
  80. // 如果设置了消息频率则验证,转发不收限制
  81. if ($sendInterval && !$isForward) {
  82. if (Cache::has('send_' . $uid)) {
  83. $this->error=lang('im.sendTimeLimit',['time'=>$sendInterval]);
  84. return false;
  85. }
  86. }
  87. if($param['type']=='text'){
  88. // 限制文字内容长度
  89. $text = strip_tags($param['content']);
  90. $textLen = mb_strlen($text);
  91. if ($textLen > 2048) {
  92. $this->error=lang('im.msgContentLimit') . $textLen;
  93. return false;
  94. }
  95. $param['content'] = preg_link($param['content']);
  96. // 接入聊天内容检测服务
  97. event('GreenText',['content'=>$param['content'],'service'=>"chat_detection"]);
  98. }
  99. $chatSetting = $globalConfig['chatInfo'];
  100. if($param['toContactId']!=-1){
  101. if ($is_group == 0) {
  102. $kefuUser=$chatSetting['autoAddUser']['user_ids'] ?? [];
  103. $manageUser=User::where([['status','=',1],['role','>',0]])->column('user_id');
  104. $kefu=array_unique(array_merge($kefuUser,$manageUser));
  105. $csUid = self::$userInfo['cs_uid'] ?? 0;
  106. $manage=false;
  107. // 发送者和接受者是客服或者管理员也可以发送消息
  108. if(in_array($uid,$kefu) || in_array($param['toContactId'],$kefu)){
  109. $manage=true;
  110. }
  111. if($chatSetting['simpleChat'] == 0 && !$manage){
  112. $this->error=lang('im.forbidChat');
  113. return false;
  114. }
  115. // 如果是单聊,并且是社区模式和不是自己的客服、需要判断是否是好友
  116. if ($globalConfig['sysInfo']['runMode'] == 2 && $csUid != $param['toContactId'] && !$manage) {
  117. // 判断我是不是对方的客服
  118. $cus = User::where(['user_id' => $param['toContactId']])->value('cs_uid');
  119. if ($cus != $uid) {
  120. $friend = Friend::where(['friend_user_id' => $uid, 'create_user' => $param['toContactId']])->find();
  121. if (!$friend) {
  122. $this->error=lang('im.notFriend');
  123. return false;
  124. }
  125. $otherFriend = Friend::where(['friend_user_id' => $param['toContactId'], 'create_user' => $uid])->find();
  126. if (!$otherFriend) {
  127. $this->error=lang('im.friendNot');
  128. return false;
  129. }
  130. }
  131. }
  132. //判断是否给机器人客服发送消息
  133. $autoTask = Config::autoTask();
  134. if (!empty($autoTask['user_id']) && $param['toContactId'] == $autoTask['user_id']) {
  135. $is_robot = true;
  136. }
  137. }else{
  138. // 群聊必须群成员才能发送消息
  139. $group_id = explode('-', $param['toContactId'])[1] ?? '';
  140. if(!$group_id){
  141. $this->error=lang('system.parameterError');
  142. return false;
  143. }
  144. if(!self::nospeak($group_id,$uid)){
  145. if($isForward){
  146. return false;
  147. }
  148. return shutdown(lang('group.notSpeak'));
  149. }
  150. // 群聊必须群成员才能发送消息
  151. $groupUser=GroupUser::where(['user_id'=>$uid,'status'=>1,'group_id'=>$group_id,'delete_time'=>0])->find();
  152. if(!$groupUser){
  153. $this->error = lang('group.notCustom');
  154. return false;
  155. }
  156. if($groupUser['no_speak_time']>time()){
  157. $this->error = lang('group.notSpeak',['time'=>date('Y-m-d H:i:s',$groupUser['no_speak_time'])]);
  158. return false;
  159. }
  160. }
  161. }
  162. if ($sendInterval) {
  163. Cache::set('send_' . $uid, time(), $sendInterval);
  164. }
  165. $data = self::sendMsg($param,$is_group, 0, $uid);
  166. // 机器人自动回复问题推送给用户
  167. if ($is_robot && $param['type'] == 'text') {
  168. if (!empty($param['question_id'])) {
  169. $param['type'] = 'answer';
  170. $param['content'] = QuestionLanguages::where('id', $param['question_id'])->value('answer');
  171. $param['extends'] = json_encode([
  172. 'question_id' => $param['question_id'],
  173. ]);
  174. } else {
  175. //获取关键词匹配
  176. $keyword_ids = KeywordLanguages::getKeywordByContent($param['content']);
  177. $question = QuestionLanguages::getQuestion($keyword_ids, $param['content']);
  178. $data['question'] = $question;
  179. if ($question) {
  180. $param['type'] = 'list';
  181. $content = [
  182. 'title' => Config::getFieldValue('reply_keyword',$param['language_code']),
  183. 'list' => $question,
  184. ];
  185. $param['content'] = json_encode($content);
  186. } else {
  187. $param['type'] = 'text';
  188. $param['content'] = Config::getFieldValue('reply_keyword_math_fail',$param['language_code']);
  189. }
  190. }
  191. $fromUserId = (int)$param['toContactId'];
  192. $user_id = (int)$param['user_id'];
  193. $param['toContactId'] = $user_id;
  194. $param['user_id'] = $fromUserId;
  195. $param['fromUser'] = null;
  196. $param['id'] = \utils\Str::getUuid();
  197. self::sendMsg($param,$is_group, 0, $user_id);
  198. //机器人自动回复消息
  199. //event('AutoReplyMessage', ['param' => $param, 'globalConfig' => $globalConfig]);
  200. }
  201. return $data;
  202. }
  203. //实际发送消息
  204. public static function sendMsg($param,$is_group=0,$is_sys=0, $uid=0){
  205. // $uid = $uid ?: ($param['user_id'] ?? 1);
  206. $uid = self::$uid ?: ($param['user_id'] ?? 1);
  207. $toContactId=$param['toContactId'];
  208. $manage=[];
  209. // 重新建立会话,更新会话删除记录
  210. $isDelChat=ChatDelog::where(['user_id'=>$uid,'to_user'=>$toContactId])->find();
  211. if($isDelChat){
  212. ChatDelog::where(['user_id'=>$uid,'to_user'=>$toContactId])->delete();
  213. ChatDelog::updateCache($uid);
  214. }
  215. if($is_group==1){
  216. $group_id = explode('-', $param['toContactId'])[1] ?? '';
  217. $chat_identify=$toContactId;
  218. $toContactId=$group_id;
  219. $manage=GroupUser::getGroupManage($group_id);
  220. }else{
  221. $chat_identify=chat_identify($param['user_id'],$toContactId);
  222. }
  223. $fileSzie=isset($param['file_size'])?$param['file_size']:'';
  224. $fileName=isset($param['file_name'])?$param['file_name']:'';
  225. $ossUrl=getDiskUrl();
  226. // 如果是转发图片文件的消息,必须把域名去除掉
  227. $content=$param['content'];
  228. if(in_array($param['type'],self::$fileType)){
  229. if(strpos($param['content'],$ossUrl)!==false){
  230. $content=str_replace($ossUrl,'',$param['content']);
  231. }
  232. }
  233. $param['content']=$content;
  234. $atList=($param['at'] ?? null) ? array_map('intval', $param['at']): [];
  235. // 如果at里面有0,代表@所有人
  236. if($atList && in_array(0,$atList)){
  237. $atList=GroupUser::where([['group_id','=',$toContactId],['status','=',1],['user_id','<>',$param['user_id']]])->column('user_id');
  238. }
  239. $at=$atList ? implode(',',$atList) : null;
  240. $data=[
  241. 'from_user'=>$param['user_id'],
  242. 'to_user'=>$toContactId,
  243. 'id'=>$param['id'],
  244. 'content'=>$param['type'] != 'list' ? str_encipher($param['content'],true) : $param['content'],
  245. 'chat_identify'=>$chat_identify,
  246. 'create_time'=>time(),
  247. 'type'=>$param['type'],
  248. 'is_group'=>$toContactId==-1 ? 3 : $is_group,
  249. 'is_read'=>$is_group ? 1 : 0,
  250. 'file_id'=>$param['file_id'] ?? 0,
  251. "file_cate"=>$param['file_cate'] ?? 0,
  252. 'file_size'=>$fileSzie,
  253. 'file_name'=>$fileName,
  254. 'at'=>$at,
  255. 'pid'=>$param['pid'] ?? 0,
  256. 'extends'=>($param['extends'] ?? null) ? $param['extends'] : null,
  257. 'translate_content' => $param['translate_content'] ?? null,
  258. 'language_code' => $param['language_code'] ?? '',
  259. ];
  260. $message=new self();
  261. $message->update(['is_last'=>0],['chat_identify'=>$chat_identify]);
  262. $message->save($data);
  263. // 拼接消息推送
  264. $type=$is_group?'group':'simple';
  265. $sendData=$param;
  266. $sendData['status']='succeed';
  267. $sendData['at']=$atList;
  268. $sendData['msg_id']=$message->msg_id;
  269. $sendData['is_click']=(int)$message->is_click;
  270. $sendData['is_read']=0;
  271. $sendData['to_user']=(string)$toContactId;
  272. $sendData['role']=$manage[self::$uid] ?? 3;
  273. $sendData['sendTime']=(int)$sendData['sendTime'];
  274. $sendData['translate_content'] = $param['translate_content'] ?? '';
  275. //这里单聊中发送对方的消息,对方是接受状态,自己是对方的联系人,要把发送对象设置为发送者的ID。
  276. if($is_group){
  277. $sendData['toContactId']=$param['toContactId'];
  278. // 将团队所有成员的未读状态+1
  279. GroupUser::editGroupUser([['group_id','=',$toContactId],['user_id','<>',$uid]],['unread'=>Db::raw('unread+1')]);
  280. }else{
  281. //$sendData['toContactId']=$toContactId;//$uid;
  282. $sendData['toContactId'] = $param['user_id'];
  283. }
  284. $sendData['fileSize']=$fileSzie;
  285. $sendData['fileName']=$fileName;
  286. if(in_array($sendData['type'],self::$fileType)){
  287. $sendData['content']=getFileUrl($sendData['content']);
  288. if($sendData['type']=='image'){
  289. $pre=1;
  290. }else{
  291. $pre=2;
  292. }
  293. $sendData['preview']=previewUrl($sendData['content'],$pre);
  294. $sendData['extUrl']=getExtUrl($sendData['content']);
  295. $sendData['download']= $sendData['file_id'] ? getMainHost().'/filedown/'.encryptIds($sendData['file_id']) : '';
  296. }
  297. $forContactId=$sendData['toContactId'];
  298. if($is_sys){
  299. $forContactId=$toContactId;
  300. }
  301. if($is_group==0){
  302. $toContactId=[$toContactId,$param['user_id']];
  303. }
  304. $sendData['toUser']=(string)$param['toContactId'];
  305. $user=new User();
  306. // 将聊天窗口的联系人信息带上,方便临时会话
  307. $sendData['contactInfo']=$user->setContact($forContactId,$is_group,$sendData['type'],$sendData['content']);
  308. if (empty($sendData['fromUser'])) {
  309. $sendData['fromUser'] = [
  310. 'id' => $sendData['contactInfo']['id'],
  311. 'displayName' => $sendData['contactInfo']['displayName'],
  312. 'avatar' => $sendData['contactInfo']['avatar'],
  313. 'account' => '',
  314. ];
  315. }
  316. $sendData['fromUser']['id']=(int)$sendData['fromUser']['id'];
  317. // 向发送方发送消息
  318. wsSendMsg($toContactId,$type,$sendData,$is_group);
  319. $sendData['toContactId']=$param['toContactId'];
  320. return $sendData;
  321. }
  322. // 群禁言
  323. public static function nospeak($group_id,$user_id){
  324. $group=Group::find($group_id);
  325. if($group->owner_id==$user_id){
  326. return true;
  327. }
  328. if($group->setting){
  329. $setting=json_decode($group->setting,true);
  330. $nospeak=isset($setting['nospeak'])?$setting['nospeak']:0;
  331. $role=GroupUser::where(['group_id'=>$group_id,'user_id'=>$user_id])->value('role');
  332. if($nospeak==1 && $role>2){
  333. return false;
  334. }elseif($nospeak==2 && $role!=1){
  335. return false;
  336. }
  337. }
  338. return true;
  339. }
  340. // 将消息中的@用户加入到atListQueue中
  341. public static function setAtread($messages,$user_id){
  342. foreach($messages as $k=>$v){
  343. if(!isset($v['at'])){
  344. continue;
  345. }
  346. if($v['at'] && in_array($user_id,$v['at'])){
  347. $atListQueue=Cache::get("atListQueue");
  348. $atListQueue[$v['msg_id']][]=$user_id;
  349. Cache::set("atListQueue",$atListQueue);
  350. }
  351. }
  352. }
  353. }