Online.php 37 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085
  1. <?php
  2. namespace addons\shopro\library\chat;
  3. use \GatewayWorker\Lib\Gateway;
  4. use addons\shopro\model\chat\Connection;
  5. use addons\shopro\model\chat\CustomerService;
  6. use addons\shopro\model\chat\Log as ChatLog;
  7. use addons\shopro\model\chat\User as ChatUser;
  8. use addons\shopro\library\chat\traits\GetLinkerTrait;
  9. use addons\shopro\model\chat\Question;
  10. use addons\shopro\model\User;
  11. use app\admin\model\Admin;
  12. class Online
  13. {
  14. use GetLinkerTrait;
  15. public static function getConfig($type) {
  16. // 初始化 workerman 的时候不能读取数据库,会导致数据库连接异常
  17. $config = require(ROOT_PATH . 'addons' . DS . 'shopro' . DS . 'library' . DS . 'chat' . DS . 'config.php');
  18. $system = $config[$type] ?? [];
  19. return $system;
  20. }
  21. /**
  22. * 先判断是否有对象存储,将对象存储配置注入到 upload 配置
  23. *
  24. * @return void
  25. */
  26. public static function uploadConfigInit() {
  27. // 获取绑定的 插件 钩子
  28. $hooks = config('addons.hooks');
  29. $upload_config_inits = isset($hooks['upload_config_init']) && $hooks['upload_config_init'] ? $hooks['upload_config_init'] : [];
  30. if ($upload_config_inits) {
  31. $storage = end($upload_config_inits);
  32. \think\Hook::add('upload_config_init', 'addons\\' . ucfirst($storage) . '\\' . ucfirst($storage));
  33. // 注入 storage 配置
  34. $upload = \app\common\model\Config::upload();
  35. \think\Hook::listen("upload_config_init", $upload);
  36. \think\Config::set('upload', array_merge(\think\Config::get('upload'), $upload));
  37. }
  38. }
  39. /**
  40. * 客服自定义 cdnurl 方法
  41. * 1、提供默认域名,如果没配置对象存储,拼接当前接口域名(访问域名的直接获取,websocket 的从 websocket SESSION['server']请求头中获取)
  42. * 2、调用默认 cdnurl 方法
  43. *
  44. * @param [type] $val
  45. * @param [type] $domain
  46. * @return void
  47. */
  48. public static function cdnurl($val, $domain = null) {
  49. $domain = $domain ? : self::getDomain();
  50. return cdnurl($val, $domain);
  51. }
  52. /**
  53. * 获取 websocket 的域名拼接 cdnurl
  54. *
  55. * @return void
  56. */
  57. public static function getDomain() {
  58. // 优先以正常访问方式获取域名
  59. $domain = request()->domain();
  60. $host = request()->host();
  61. if (!$domain || !$host) {
  62. // 如果不能获取,说明是 websocket 连接,获取当前 server 中的 domain
  63. $server = $_SESSION['server'];
  64. $systemConfig = self::getConfig('system');
  65. $is_ssl = $systemConfig['is_ssl'] ? true : false;
  66. $domain = ($is_ssl ? 'https://' : 'http://') . $server['SERVER_NAME'];
  67. }
  68. return $domain;
  69. }
  70. // 主要为了记录当前系统总共有多少种分组
  71. public static function getGrouponName($type, $data = []) {
  72. switch($type) {
  73. case 'online_user' : // 当前在线用户分组
  74. $group_name = 'online_user';
  75. break;
  76. case 'online_waiting' : // 当前在线用户待分配客服分组
  77. $group_name = 'online_waiting';
  78. break;
  79. case 'online_customer_service' : // 当前在线客服数组
  80. $group_name = 'online_customer_service';
  81. break;
  82. case 'customer_service_user': // 当前在线用户所在的客服分组
  83. $group_name = 'customer_service_user:' . ($data['customer_service_id'] ?? 0);
  84. break;
  85. default :
  86. $group_name = $type;
  87. break;
  88. }
  89. return $group_name;
  90. }
  91. public static function getUId($id, $type)
  92. {
  93. $ids = is_array($id) ? $id : [$id];
  94. foreach ($ids as &$i) {
  95. $i = $type . '-' . $i;
  96. }
  97. return is_array($id) ? $ids : $ids[0];
  98. }
  99. public static function updateChatUser($session_id, $user) {
  100. $chatUser = ChatUser::where(function ($query) use ($session_id, $user) {
  101. $query->where('session_id', $session_id)
  102. ->whereOr(function ($query) use ($user) {
  103. $query->where('user_id', '<>', 0)
  104. ->where('user_id', ($user ? $user['id'] : 0));
  105. });
  106. })->find();
  107. $defaultAvatar = null;
  108. $config = \addons\shopro\model\Config::where('name', 'user')->find();
  109. if ($config) {
  110. $userConfig = json_decode($config['value'], true);
  111. $defaultAvatar = $userConfig['avatar'];
  112. }
  113. if (!$chatUser) {
  114. $chatUser = new ChatUser();
  115. $chatUser->session_id = $session_id;
  116. $chatUser->user_id = $user ? $user['id'] : 0;
  117. $chatUser->nickname = $user ? $user['nickname'] : ('游客-' . substr($session_id, 0, 5));
  118. $chatUser->avatar = $user ? $user->getData('avatar') : $defaultAvatar;
  119. $chatUser->customer_service_id = 0; // 断开连接的时候存入
  120. $chatUser->lasttime = time();
  121. } else {
  122. if ($user) {
  123. // 更新用户信息
  124. $chatUser->user_id = $user['id'] ?? 0;
  125. $chatUser->nickname = $user['nickname'] ? $user['nickname'] : ('游客-' . substr($session_id, 0, 5));
  126. $chatUser->avatar = $user['avatar'] ? $user->getData('avatar') : $defaultAvatar;
  127. }
  128. $chatUser->lasttime = time(); // 更新时间
  129. }
  130. $chatUser->save();
  131. return $chatUser->toArray();
  132. }
  133. /**
  134. * 根据 id 获取客服
  135. */
  136. public static function customerServiceById($customer_service_id)
  137. {
  138. $customerService = CustomerService::where('id', $customer_service_id)->find();
  139. return $customerService;
  140. }
  141. /**
  142. * 检查客服身份
  143. *
  144. * @param string $token
  145. * @param string $customer_service_id
  146. * @param string $expire_time
  147. * @return array
  148. */
  149. public static function checkAdmin($token, $customer_service_id, $expire_time) {
  150. if ($token && $customer_service_id) {
  151. // 获取客服信息
  152. $customerService = self::customerServiceById($customer_service_id);
  153. if (!$customerService) {
  154. return false;
  155. }
  156. $customerService = $customerService->toArray();
  157. // 获取 admin
  158. $admin = Admin::get($customerService['admin_id']);
  159. if (!$admin) {
  160. return false;
  161. }
  162. $admin = $admin->toArray();
  163. $current_token = md5($admin['username'] . $expire_time);
  164. if (($expire_time + (86400 * 30)) > time() && $current_token == $token) {
  165. // 验证通过
  166. return [
  167. 'customerService' => $customerService,
  168. 'admin' => $admin
  169. ];
  170. }
  171. }
  172. return false;
  173. }
  174. /**
  175. * 检测并获取用户
  176. *
  177. * @param [type] $token
  178. * @return void
  179. */
  180. public static function checkUser($token) {
  181. $data = \app\common\library\Token::get($token);
  182. if (!$data) {
  183. return false;
  184. }
  185. $user_id = intval($data['user_id']);
  186. if ($user_id <= 0) {
  187. return false;
  188. }
  189. // 这个 user 不能转数组,用到了 $user->getData('avatar') 去拿原始的未拼接 cdnurl 头像地址
  190. $user = User::where('id', $user_id)->find();
  191. if (!$user) {
  192. return false;
  193. }
  194. return $user;
  195. }
  196. /**
  197. * 判断客服在线状态, 只能客服用
  198. *
  199. * @param [type] $customer_service_id
  200. * @return void
  201. */
  202. public static function customerServiceStatusById($customer_service_id, $oper_type = 'customer_service', $data = [])
  203. {
  204. $current_client_id = $data['client_id'] ?? ''; // oper_type = customer_service 时存在
  205. // 获取当前用户的所有session,并且当前连接的session是最新的
  206. $customerServiceSessions = self::onlineCustomerServiceNewSessionById($customer_service_id, $current_client_id);
  207. $status = false;
  208. foreach ($customerServiceSessions as $key => $customerServiceSession) {
  209. $customerService = $customerServiceSession['user'] ?? [];
  210. if ($customerService && in_array($customerService['status'], ['online', 'busy'])) {
  211. // 只要其中一个在线,即为在线
  212. $status = true;
  213. }
  214. }
  215. return $status;
  216. }
  217. /**
  218. * 客服上线
  219. *
  220. * @param string $client_id
  221. * @param object $customerService
  222. * @return object
  223. */
  224. public static function customerServiceOnline($client_id, $customerService)
  225. {
  226. // 更新客服状态
  227. CustomerService::where('id', $customerService['id'])->update([
  228. 'status' => 'online'
  229. ]);
  230. // 重新更新客服信息
  231. $customerService['status'] = 'online';
  232. // 更新客服 session 为 online
  233. $_SESSION['user']['status'] = 'online';
  234. // 加入在线客服组
  235. Gateway::joinGroup($client_id, Online::getGrouponName('online_customer_service'));
  236. // 通知客服连接成功
  237. Sender::customerServiceInit($client_id, $customerService);
  238. // 通知连接的用户,客服上线了
  239. Sender::customerServiceOnline($customerService);
  240. // 通知所有在线客服,更新当前在线客服列表
  241. Sender::customerServiceOnlineList();
  242. return $customerService;
  243. }
  244. /**
  245. * 客服忙碌
  246. *
  247. * @param string $client_id
  248. * @param object $customerService
  249. * @return object
  250. */
  251. public static function customerServiceBusy($client_id, $customerService)
  252. {
  253. // 更新客服状态
  254. CustomerService::where('id', $customerService['id'])->update([
  255. 'status' => 'busy'
  256. ]);
  257. // 重新更新客服信息
  258. $customerService['status'] = 'busy';
  259. // 更新客服 session 为 busy
  260. $_SESSION['user']['status'] = 'busy';
  261. return $customerService;
  262. }
  263. /**
  264. * 客服下线(这里不更新数据库,并且只更新当前连接为下线,如果当前客服在别的浏览器也有登录,则不受此操作影响)
  265. *
  266. * @param string $client_id
  267. * @param object $customerService
  268. * @return object
  269. */
  270. public static function customerServiceOffline($client_id = null, $customerService)
  271. {
  272. // 重新更新客服信息
  273. $customerService['status'] = 'offline';
  274. // 更新客服 session 为 offline
  275. $_SESSION['user']['status'] = 'offline';
  276. // 移除在线客服
  277. Gateway::leaveGroup($client_id, Online::getGrouponName('online_customer_service'));
  278. // 获取客服在线状态
  279. $onlineStatus = self::customerServiceStatusById($customerService['id'], 'customer_service', ['client_id' => $client_id]);
  280. if (!$onlineStatus) {
  281. // 绑定该 uid 的所有 session 的客服状态都离线了或者客服真实离线了,通知用户客服下线
  282. Sender::customerServiceOffline($customerService);
  283. }
  284. // 通知所有客服更新在线的客服列表
  285. Sender::customerServiceOnlineList();
  286. return $customerService;
  287. }
  288. /**
  289. * 客服真实离线
  290. *
  291. * @param int $customer_service_id
  292. * @param array $customerService
  293. * @return void
  294. */
  295. public static function customerServiceRealOffline($customer_service_id, $customerService) {
  296. $isUidOnline = Gateway::isUidOnline(online::getUId($customer_service_id, 'customer_service'));
  297. if (!$isUidOnline) {
  298. // 绑定该 uid 的所有 client_id 都离线了,更新客服数据库为离线状态
  299. CustomerService::where('id', $customerService['id'])->update([
  300. 'status' => 'offline'
  301. ]);
  302. }
  303. }
  304. /**
  305. * 更新客服最后接入时间
  306. *
  307. * @param [type] $customerService
  308. * @param [type] $oper_type
  309. * @return void
  310. */
  311. public static function updateCustomerServiceLasttime($customerService, $oper_type)
  312. {
  313. // 更新 session 的 lasttime
  314. if ($oper_type == 'customer_service') {
  315. // 如果是客服自己操作接入
  316. $_SESSION['user']['lasttime'] = time();
  317. } else {
  318. // 获取客服 client_id;
  319. $customerServiceClientIds = Gateway::getClientIdByUid(Online::getUId($customerService['id'], 'customer_service'));
  320. foreach ($customerServiceClientIds as $customer_service_client_id) {
  321. $session = Gateway::getSession($customer_service_client_id);
  322. $customerService = $session['user']; // 客服信息
  323. $customerService['lasttime'] = time(); // 更新最后接入时间
  324. // 更新 session 缓存
  325. Gateway::updateSession($customer_service_client_id, [
  326. 'user' => $customerService,
  327. ]);
  328. }
  329. }
  330. // 更新数据库
  331. CustomerService::where('id', $customerService['id'])->update([
  332. 'lasttime' => time()
  333. ]);
  334. }
  335. /**
  336. * 用户最后一次的链接
  337. *
  338. * @param integer $user_id
  339. * @param string $session_id
  340. * @return array
  341. */
  342. public static function userLastConnection($user_id = 0, $session_id = '') {
  343. $lastConnection = Connection::where(function ($query) use ($user_id, $session_id) {
  344. if ($session_id) {
  345. $query->where(function ($query) use ($session_id) {
  346. $query->where('session_id', '<>', '')
  347. ->where('session_id', 'not null')
  348. ->where('session_id', $session_id);
  349. });
  350. }
  351. if ($user_id) {
  352. $query->where(function ($query) use ($user_id) {
  353. $query->where('user_id', '<>', 0)
  354. ->where('user_id', 'not null')
  355. ->where('user_id', $user_id);
  356. });
  357. }
  358. })->where('customer_service_id', '<>', 0)->order('id', 'desc')->find();
  359. return $lastConnection;
  360. }
  361. /**
  362. * 关闭当前 session_id 的所有连接
  363. */
  364. public static function closeConnectionBySessionId($session_id)
  365. {
  366. Connection::where('session_id', $session_id)->where('status', 'in', ['ing', 'waiting'])->update([
  367. 'status' => 'end'
  368. ]);
  369. }
  370. /**
  371. * 分配客服
  372. *
  373. * @param string $session_id
  374. * @param string $user_id
  375. * @return array
  376. */
  377. public static function allocatCustomerService($session_id, $user_id) {
  378. $config = self::getConfig('basic');
  379. $last_customer_service = $config['last_customer_service'] ?? 1;
  380. $allocate = $config['allocate'] ?? 'busy';
  381. // 分配的客服
  382. $currentCustomerService = null;
  383. // 使用上次客服
  384. if ($last_customer_service) {
  385. // 获取上次连接的信息
  386. $lastConnection = self::userLastConnection($user_id, $session_id);
  387. $lastCustomerService = null;
  388. $currentCustomerService = null;
  389. if ($lastConnection) {
  390. // 获取上次客服信息
  391. // $lastCustomerService = Online::customerServiceById($lastConnection['customer_service_id']); // 读取数据库,不准确
  392. $lastCustomerService = Self::onlineCustomerServiceById($lastConnection['customer_service_id']); // 获取socket 连接里面上次客服是否在线
  393. // 判断客服是否在线
  394. if ($lastCustomerService) {
  395. if ($lastCustomerService['status'] == 'online') {
  396. $currentCustomerService = $lastCustomerService;
  397. }
  398. }
  399. }
  400. }
  401. // 没有客服,随机分配
  402. if (!$currentCustomerService) {
  403. // 在线客服列表
  404. $onlineCustomerServices = self::onlineCustomerServices();
  405. if ($onlineCustomerServices) {
  406. if ($allocate == 'busy') {
  407. // 将客服列表,按照工作繁忙程度正序排序, 这里没有离线的客服
  408. $onlineCustomerServices = array_column($onlineCustomerServices, null, 'busy_percent');
  409. ksort($onlineCustomerServices);
  410. // 取忙碌度最小,并且客服为 正常在线状态
  411. foreach ($onlineCustomerServices as $customerService) {
  412. if ($customerService['status'] == 'online') {
  413. $currentCustomerService = $customerService;
  414. break;
  415. }
  416. }
  417. if (!$currentCustomerService) {
  418. // 如果都不是 online 状态,默认取第一条
  419. $currentCustomerService = $onlineCustomerServices[0] ?? null;
  420. }
  421. } else if ($allocate == 'turns') {
  422. // 按照最后接入时间正序排序,这里没有离线的客服
  423. $onlineCustomerServices = array_column($onlineCustomerServices, null, 'lasttime');
  424. ksort($onlineCustomerServices);
  425. // 取忙碌度最小,并且客服为 正常在线状态
  426. foreach ($onlineCustomerServices as $customerService) {
  427. if ($customerService['status'] == 'online') {
  428. $currentCustomerService = $customerService;
  429. break;
  430. }
  431. }
  432. if (!$currentCustomerService) {
  433. // 如果都不是 online 状态,默认取第一条
  434. $currentCustomerService = $onlineCustomerServices[0] ?? null;
  435. }
  436. } else if ($allocate == 'random') {
  437. // 随机获取一名客服
  438. $onlineCustomerServices = array_column($onlineCustomerServices, null, 'id');
  439. $customer_service_id = 0;
  440. if ($onlineCustomerServices) {
  441. $customer_service_id = array_rand($onlineCustomerServices);
  442. }
  443. $currentCustomerService = $onlineCustomerServices[$customer_service_id] ?? null;
  444. }
  445. }
  446. }
  447. return $currentCustomerService;
  448. }
  449. /**
  450. * 通过 session_id 记录客服信息,并且加入对应的客服组
  451. *
  452. * @param string $session_id 用户
  453. * @param array $customerService 客服信息
  454. * @return null
  455. */
  456. public static function bindCustomerServiceBySessionId($session_id, $customerService, $oper_type = 'customer_service') {
  457. $uid = Online::getUId($session_id, 'user');
  458. $client_ids = Gateway::getClientIdByUid($uid);
  459. // 将当前 session_id 绑定的 client_id 都加入当前客服组
  460. foreach ($client_ids as $client_id) {
  461. self::bindCustomerService($client_id, $customerService, $oper_type);
  462. }
  463. }
  464. /**
  465. * 记录客服信息,并且加入对应的客服组
  466. *
  467. * @param string $client_id 用户
  468. * @param array $customerService 客服信息
  469. * @return null
  470. */
  471. public static function bindCustomerService($client_id, $customerService, $oper_type = 'user') {
  472. // 当前用户使用 updateSession 会吧之前的内容覆盖掉,并且 getSession 无法获取刚刚设置的 session
  473. // 更新用户的客服信息
  474. if ($oper_type == 'user') {
  475. // 当前连接者
  476. $_SESSION['customer_service_id'] = $customerService['id'];
  477. $_SESSION['customer_service'] = $customerService;
  478. } else {
  479. // 其他连接着
  480. Gateway::updateSession($client_id, [
  481. 'customer_service_id' => $customerService['id'],
  482. 'customer_service' => $customerService
  483. ]);
  484. }
  485. // 更新客服的最后接入用户时间
  486. self::updateCustomerServiceLasttime($customerService, $oper_type);
  487. // 加入对应客服组,统计客服信息 customer_service_user:客服 ID
  488. Gateway::joinGroup($client_id, Online::getGrouponName('customer_service_user', ['customer_service_id' => $customerService['id']]));
  489. // 从等待接入组移除
  490. Gateway::leaveGroup($client_id, Online::getGrouponName('online_waiting'));
  491. }
  492. /**
  493. * 通过 session_id 将用户移除客服组
  494. *
  495. * @param string $session_id 用户
  496. * @param array $customerService 客服信息
  497. * @return null
  498. */
  499. public static function unBindCustomerServiceBySessionId($session_id, $customerService, $oper_type = 'customer_service')
  500. {
  501. $uid = Online::getUId($session_id, 'user');
  502. $client_ids = Gateway::getClientIdByUid($uid);
  503. // 将当前 session_id 绑定的 client_id 都移除当前客服组
  504. foreach ($client_ids as $client_id) {
  505. // 这里新客服已经绑定了,不需要移除session 了,所以 remove_session 为 false
  506. self::unBindCustomerService($client_id, $customerService, false, $oper_type);
  507. }
  508. }
  509. /**
  510. * 将用户的客服信息移除
  511. *
  512. * @param string $client_id 用户
  513. * @param array $customerService 客服信息
  514. * @return null
  515. */
  516. public static function unBindCustomerService($client_id, $customerService, $remove_session = false, $oper_type = false)
  517. {
  518. if ($remove_session) {
  519. if ($oper_type == 'user') {
  520. // 当前连接者
  521. $_SESSION['customer_service_id'] = 0;
  522. $_SESSION['customer_service'] = [];
  523. } else {
  524. // 其他连接着
  525. Gateway::updateSession($client_id, [
  526. 'customer_service_id' => 0,
  527. 'customer_service' => []
  528. ]);
  529. }
  530. }
  531. // 移除对应客服组
  532. Gateway::leaveGroup($client_id, Online::getGrouponName('customer_service_user', ['customer_service_id' => $customerService['id']]));
  533. }
  534. // 转接用户
  535. public static function transferCustomerServiceBySessionId($session_id, $newCustomerService, $oldCustomerService, $oper_type = 'customer_service') {
  536. // 将老客服的服务记录直接保存
  537. $chatUser = Self::getChatUserBySessionId($session_id);
  538. if ($chatUser && $chatUser->user_id) {
  539. $user = User::get($chatUser->user_id);
  540. }
  541. $userData = [
  542. 'user' => $user ?? null,
  543. 'session_id' => $session_id,
  544. 'chat_user' => $chatUser
  545. ];
  546. // 创建并结束 connection
  547. $connection = Online::checkOrCreateConnection($userData, $oldCustomerService, true);
  548. // 新客服接入用户
  549. self::bindCustomerServiceBySessionId($session_id, $newCustomerService, $oper_type);
  550. // 将用户从旧的客服组移除
  551. self::unBindCustomerServiceBySessionId($session_id, $oldCustomerService, $oper_type);
  552. }
  553. /**
  554. * 排队的用户,将用户加入 waiting 组
  555. *
  556. * @param string $client_id
  557. * @return null
  558. */
  559. public static function bindWaiting ($client_id) {
  560. Gateway::joinGroup($client_id, Online::getGrouponName('online_waiting'));
  561. }
  562. /**
  563. * 判断是否有链接,如果没有创建新链接(为了避免用户刷新,然后这里重新创建新纪录)
  564. */
  565. public static function checkOrCreateConnection($userData, $customerService, $set_end = false) {
  566. $user = $userData['user'] ?? null;
  567. $chatUser = $userData['chat_user'] ?? null;
  568. $session_id = $userData['session_id'] ?? '';
  569. // 正在进行中的连接
  570. $ingConnection = Connection::where('customer_service_id', $customerService['id'])
  571. ->where('status', 'in', ['ing', 'waiting'])
  572. ->where(function ($query) use ($user, $session_id) {
  573. $query->where('session_id', $session_id)->whereOr('user_id', ($user->id ?? 0));
  574. })->find();
  575. if (!$ingConnection) {
  576. // 不存在,创建新的
  577. $ingConnection = new Connection();
  578. $ingConnection->user_id = $user ? $user['id'] : 0;
  579. $ingConnection->nickname = $chatUser['nickname'];
  580. $ingConnection->session_id = $session_id;
  581. $ingConnection->customer_service_id = $customerService ? $customerService['id'] : 0; // 0 没有客服在;
  582. $ingConnection->starttime = time();
  583. $ingConnection->endtime = $set_end ? time() : 0;
  584. $ingConnection->status = $set_end ? 'end' : ($customerService ? 'ing' : 'waiting');
  585. $ingConnection->createtime = time();
  586. $ingConnection->updatetime = time();
  587. $ingConnection->save();
  588. } else {
  589. if ($ingConnection->status == 'waiting' && $customerService) {
  590. // 如果是 waiting, 并且存在客服,修改为 ing
  591. $ingConnection->customer_service_id = $customerService['id'];
  592. $ingConnection->status = 'ing';
  593. }
  594. if ($set_end) {
  595. $ingConnection->status = 'end';
  596. $ingConnection->endtime = time();
  597. }
  598. $ingConnection->save();
  599. }
  600. return $ingConnection;
  601. }
  602. /**
  603. * 用户登录,更新当前 session_id 没有 user_id 的记录
  604. */
  605. public static function sessionUserSave($user, $session_id) {
  606. // 更新用户连接
  607. $connection = Connection::where('session_id', $session_id)->where('user_id', 0)->update([
  608. 'user_id' => $user['id'],
  609. 'nickname' => $user['nickname']
  610. ]);
  611. // 更新用户消息记录
  612. $connection = ChatLog::where('session_id', $session_id)->where('user_id', 0)->update([
  613. 'user_id' => $user['id']
  614. ]);
  615. return true;
  616. }
  617. /**
  618. * 通过连接 id 获取连接
  619. */
  620. public static function connectionById($connection_id = 0) {
  621. $connection = Connection::where('id', $connection_id)->find();
  622. return $connection;
  623. }
  624. /**
  625. * 通过客服 id 获取当前客服正在服务的客户
  626. */
  627. public static function onlineByCustomerServiceId($customerService) {
  628. $onlines = Connection::where('status', 'ing')->where('customer_service_id', $customerService['id'])->select();
  629. return $onlines;
  630. }
  631. // 通过客服 id 获取当前客服历史服务过的客户
  632. public static function historyByCustomerServiceId($customerService, $data)
  633. {
  634. $except = $data['except'] ?? [];
  635. // 关闭 sql mode 的 ONLY_FULL_GROUP_BY
  636. $oldModes = closeStrict(['ONLY_FULL_GROUP_BY']);
  637. $historieLogs = Connection::with(['chat_user'])
  638. ->where('customer_service_id', $customerService['id'])
  639. ->where('session_id', 'not in', $except)
  640. ->group('session_id')
  641. ->select();
  642. // 恢复 sql mode
  643. recoverStrict($oldModes);
  644. $histories = array_column($historieLogs, 'chat_user');
  645. return array_values(array_filter($histories));
  646. }
  647. // 获取当前所有正在排队的用户
  648. public static function waiting()
  649. {
  650. $waiting = Connection::where('status', 'waiting')->select();
  651. return $waiting;
  652. }
  653. /**
  654. * 通过 session_id 删除客户服务记录
  655. *
  656. * @param array $customerService
  657. * @param string $session_id
  658. * @return void
  659. */
  660. public static function delUserBySessionId($customerService, $session_id) {
  661. return Connection::where('customer_service_id', $customerService['id'])->where('session_id', $session_id)->delete();
  662. }
  663. /**
  664. * 删除所有客户服务记录
  665. *
  666. * @param array $customerService
  667. * @param string $session_id
  668. * @return void
  669. */
  670. public static function delAllUserBySessionId($customerService)
  671. {
  672. return Connection::where('customer_service_id', $customerService['id'])->delete();
  673. }
  674. /**
  675. * 发送消息
  676. *
  677. * @param string $name 要调用的方法名,这里未使用
  678. * @param string $arguments 给要调用的方法的参数
  679. * @return object
  680. */
  681. public static function addMessage($name, $arguments) {
  682. $receive_id = $arguments[0] ?? 0;
  683. $content = $arguments[1] ?? [];
  684. $params = $arguments[2] ?? []; // 额外参数
  685. $sender = $params['sender'] ?? [];
  686. // 判断是否传入的发送人
  687. if ($sender) {
  688. // 传入了发送人信息
  689. extract($sender);
  690. } else {
  691. // 默认对发 用户 发送给客服, 客服发送给用户
  692. $sender_identify = $_SESSION['identify'];
  693. if ($sender_identify == 'customer_service') {
  694. // 获取发送信息
  695. extract(self::getCustomerServiceSenderData($receive_id, $content));
  696. } else {
  697. // 用户
  698. $session_id = $_SESSION['uid'];
  699. $user_id = $_SESSION['user_id'] ?? 0; // 如果是游客,这里为 0
  700. $user = $_SESSION['user'];
  701. $sender_id = $_SESSION['chat_user'] ? $_SESSION['chat_user']['id'] : 0;
  702. }
  703. }
  704. // 返回 message_type message
  705. extract(self::getMessageData($content));
  706. $chatLog = new ChatLog();
  707. $chatLog->session_id = $session_id;
  708. $chatLog->user_id = $user_id;
  709. $chatLog->sender_identify = $sender_identify;
  710. $chatLog->sender_id = $sender_id;
  711. $chatLog->message_type = $message_type;
  712. $chatLog->message = $message;
  713. $chatLog->save();
  714. // 加载消息人
  715. $chatLog->identify = $chatLog->identify;
  716. return $chatLog;
  717. }
  718. /**
  719. * 客服给用户发消息是,获取发送参数
  720. *
  721. * @param [type] $receive_id
  722. * @param [type] $content
  723. * @return array
  724. */
  725. public static function getCustomerServiceSenderData($receive_id, $content) {
  726. // 客服
  727. $sender_id = $customer_service_id = $_SESSION['uid'];
  728. $customerService = $_SESSION['user'];
  729. $session_id = $receive_id;
  730. $user_id = 0;
  731. $uid = Online::getUId($session_id, 'user');
  732. if (Gateway::isUidOnline($uid)) {
  733. // 接受者在线, 通过 uid 获取 client_id, 返回的是一个数组,取第一个,只是为了取对应的 user_id
  734. $client_ids = Gateway::getClientIdByUid($uid);
  735. // 获取数组第一个
  736. $client_ids = array_values(array_filter($client_ids));
  737. $client_id = $client_ids[0] ?? 0;
  738. $receiveSession = Gateway::getSession($client_id);
  739. if ($receiveSession && $receiveSession['user_id']) {
  740. $user_id = $receiveSession['user_id'];
  741. }
  742. } else {
  743. // 通过 chatUser 获取 user_id
  744. $chatUser = self::getChatUserBySessionId($session_id);
  745. if ($chatUser) {
  746. $user_id = $chatUser['user_id'];
  747. }
  748. }
  749. return compact("session_id", "user_id", "sender_id");
  750. }
  751. /**
  752. * 获取消息类型
  753. *
  754. * @param [type] $content
  755. * @return array
  756. */
  757. public static function getMessageData ($content) {
  758. $type = $content['type'] ?? '';
  759. $msg = $content['msg'] ?? '';
  760. $data = $content['data'] ?? []; // 特定 type 类型存在,包含 type = message
  761. $messageData = $data['message'] ?? []; // type=message 存在
  762. if (in_array($type, ['init', 'access'])) {
  763. // 系统消息
  764. $message_type = 'system';
  765. $message = $msg;
  766. } else if ($type == 'message') {
  767. $message_type = $messageData['message_type'] ?? 'text';
  768. $message_content = $messageData['message'] ?? '';
  769. switch ($message_type) {
  770. case 'text':
  771. $message = $message_content;
  772. break;
  773. default:
  774. $message = is_array($message_content) ? json_encode($message_content) : $message_content;
  775. break;
  776. }
  777. }
  778. return compact("message_type", "message");
  779. }
  780. /**
  781. * 将消息标记为已读
  782. *
  783. * @param array $linker 用户信息,user_id session_id
  784. * @param string $select_identify user & customer_service
  785. * @return void
  786. */
  787. public static function readMessage($linker, $select_identify = 'user')
  788. {
  789. $session_id = $linker['session_id'] ?? '';
  790. $user_id = $linker['user_id'] ?? '';
  791. $select_identify = camelize($select_identify); // 处理为 驼峰
  792. ChatLog::where(function ($query) use ($session_id, $user_id) {
  793. $query->where(function ($query) use ($session_id) {
  794. $query->where('session_id', $session_id)
  795. ->where('session_id', 'not null')
  796. ->where('session_id', '<>', '');
  797. })
  798. ->whereOr(function ($query) use ($user_id) {
  799. $query->where('user_id', $user_id)
  800. ->where('user_id', '<>', 0);
  801. });
  802. })->{$select_identify}()->update([
  803. 'readtime' => time()
  804. ]);
  805. }
  806. /**
  807. * 获取消息列表
  808. *
  809. * @param array $linker 要获取的用户
  810. * @param array $data 参数
  811. * @return array
  812. */
  813. public static function messageList($linker, $data) {
  814. $session_id = $linker['session_id'] ?? '';
  815. $user_id = $linker['user_id'] ?? '';
  816. $page = $data['page'] ?? 1;
  817. $per_page = $data['per_page'] ?? 20;
  818. $last_id = $data['last_id'] ?? 0;
  819. $messageList = ChatLog::where(function ($query) use ($session_id, $user_id) {
  820. $query->where(function ($query) use ($session_id) {
  821. $query->where('session_id', $session_id)
  822. ->where('session_id', 'not null')
  823. ->where('session_id', '<>', '');
  824. })
  825. ->whereOr(function ($query) use ($user_id) {
  826. $query->where('user_id', $user_id)
  827. ->where('user_id', '<>', 0);
  828. });
  829. });
  830. // 避免消息重复
  831. if ($last_id) {
  832. $messageList = $messageList->where('id', '<=', $last_id);
  833. }
  834. $messageList = $messageList->order('id', 'desc')->paginate($per_page, false, [
  835. 'page' => $page
  836. ]);
  837. $messageData = $messageList->items();
  838. if ($messageData) {
  839. // 批量处理消息发送人
  840. $messageData = ChatLog::formatIdentify($messageData);
  841. $messageList->data = $messageData;
  842. }
  843. return $messageList;
  844. }
  845. /**
  846. * 用户列表增加最后一条聊天记录
  847. * @param array $chatUsers
  848. * @param string $select_identify 查询对象,默认客服查用户的,用户查客服的
  849. * @return array
  850. */
  851. public static function userSetMessage($chatUsers, $select_identify = 'user') {
  852. foreach ($chatUsers as &$chatUser) {
  853. $chatUser['last_message'] = ChatLog::where('session_id', $chatUser['session_id'])->order('id', 'desc')->find();
  854. $chatUser['message_unread_num'] = ChatLog::where('session_id', $chatUser['session_id'])->where('readtime', 'null')->{$select_identify}()->count();
  855. }
  856. return $chatUsers;
  857. }
  858. public static function userRealOffline($session_id, $customerService) {
  859. // 判断 uid 是否还在线,如果是刷新浏览器,只会出现很短暂的掉线,这里检测依然还是在线状态
  860. $isUidOnline = Gateway::isUidOnline(online::getUId($session_id, 'user'));
  861. if (!$isUidOnline) {
  862. // 不在线,关闭这个用户所有连接
  863. self::closeConnectionBySessionId($session_id);
  864. // 记录用户本次客服人员,并且更新最后服务时间
  865. $chatUser = self::getChatUserBySessionId($session_id);
  866. $chatUser->customer_service_id = $customerService['id'] ?? 0;
  867. $chatUser->lasttime = time();
  868. $chatUser->save();
  869. }
  870. }
  871. /**
  872. * 常见问题
  873. *
  874. * @param int $question_id
  875. * @param array $receives
  876. * @return void
  877. */
  878. public static function questionReply ($question_id, $receives) {
  879. $question = Question::get($question_id);
  880. if ($question) {
  881. Sender::questionReply($question, $receives);
  882. }
  883. }
  884. }