Easemob.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  1. <?php
  2. namespace app\api\controller;
  3. use app\common\controller\Api;
  4. use Redis;
  5. use think\Db;
  6. /**
  7. * 环信接口
  8. */
  9. class Easemob extends Api
  10. {
  11. protected $noNeedLogin = ["callback"];
  12. protected $noNeedRight = ['*'];
  13. protected $appKey;
  14. protected $redis;
  15. public function __construct() {
  16. //初始化配置
  17. $easemob_config = config('easemob');
  18. $this->appKey = $easemob_config['appkey'];
  19. //redis
  20. $redis = new Redis();
  21. $redisconfig = config("redis");
  22. $redis->connect($redisconfig["host"], $redisconfig["port"]);
  23. if ($redisconfig['redis_pwd']) {
  24. $redis->auth($redisconfig['redis_pwd']);
  25. }
  26. if($redisconfig['redis_selectdb'] > 0){
  27. $redis->select($redisconfig['redis_selectdb']);
  28. }
  29. $this->redis = $redis;
  30. }
  31. /**
  32. * 回调
  33. * https://docs-im-beta.easemob.com/document/server-side/callback_configurations.html
  34. *
  35. * 群组和聊天室操作:
  36. *
  37. * muc:kick 踢出聊天室
  38. * {"callId":"1101231101159883#demo_1215088856588617164","channel_channel":"1101231101159883#demo_9@easemob.com","eventType":"chat","channel_user":"1101231101159883#demo_231181574537218@admin.conference.easemob.com","chat_type":"muc","security":"83f9462c7769df1c979fd9dea88e7400","is_downgrade":false,"content_type":"muc:kick","payload":{"muc_id":"1101231101159883#demo_231181574537218@conference.easemob.com","reason":"chatroom kick offline user","is_chatroom":true,"operation":"kick","status":{"description":"","error_code":"ok"}},"group_id":"231181574537218","writed_channel":false,"host":"msync@ebs-ali-beijing-msync71","appkey":"1101231101159883#demo","from":"1101231101159883#demo_9@easemob.com","to":"9","msg_id":"1215088856588617164","timestamp":1700474720591}
  39. *
  40. * muc:absence 有成员离开了聊天室
  41. * {"callId":"1101231101159883#demo_1215088856710252128","channel_channel":"1101231101159883#demo_231181574537218@admin.conference.easemob.com","eventType":"chat","channel_user":"admin@easemob.com","chat_type":"muc","security":"ec7d29ea415ea5930c2236089dff3bc2","is_downgrade":false,"content_type":"muc:absence","payload":{"muc_id":"1101231101159883#demo_231181574537218@conference.easemob.com","is_chatroom":true,"operation":"absence"},"group_id":"231181574537218","writed_channel":false,"host":"msync@ebs-ali-beijing-msync108","appkey":"1101231101159883#demo","from":"1101231101159883#demo_9@easemob.com","to":"231181574537218","msg_id":"1215088856710252128","timestamp":1700474720619}
  42. *
  43. * muc:leave 成员主动退出聊天室
  44. *
  45. *
  46. * 用户登入
  47. * {"callId":"1101231101159883#demo_927e28f2-0e88-4a58-855b-c8d26949ea4f","reason":"login","security":"6b793f812b5ed74f62d3a95d9ea48c8f","os":"android","ip":"182.37.138.94:39768","host":"msync@ebs-ali-beijing-msync60","session_id":"1215076057950983584","appkey":"1101231101159883#demo","user":"1101231101159883#demo_2@easemob.com/android_f9f51032-3369-4f02-95bb-0cbf41b26837","version":"4.1.2","timestamp":1700471740680,"status":"online"}
  48. *
  49. * 用户登出
  50. * {"callId":"1101231101159883#demo_ee82832b-3455-472d-9024-3706b1e519a7","reason":"logout","security":"6266f2a31bc3b603d8458f3d39a49d47","os":"android","ip":"182.37.138.94:39768","host":"msync@ebs-ali-beijing-msync60","session_id":"1215076057950983584","appkey":"1101231101159883#demo","user":"1101231101159883#demo_2@easemob.com/android_f9f51032-3369-4f02-95bb-0cbf41b26837","version":"4.1.2","timestamp":1700472505682,"status":"offline"}
  51. *
  52. * 用户登出(被其他设备踢掉)
  53. * {"callId":"1101231101159883#demo_b2cdc4d9-a499-4e46-8aff-4503fb83a5c5","reason":"replaced","security":"1f4bb2ebb99c2f5da20048c7ca87f352","os":"android","ip":"182.37.138.94:41044","host":"msync@ebs-ali-beijing-msync62","appkey":"1101231101159883#demo","user":"1101231101159883#demo_9@easemob.com/android_fb260fdf-3935-4499-a09d-b270fdd88a0e","version":"4.1.2","timestamp":1700473675582,"status":"offline"}
  54. */
  55. public function callback() {
  56. $this->notify_log_start();
  57. $input = file_get_contents("php://input"); // 主题信息
  58. $input = json_decode($input,true);
  59. //验证加密
  60. //解析数据格式
  61. //用户登入
  62. if(isset($input['reason']) && $input['reason'] == 'login' && isset($input['status']) && $input['status'] == 'online'){
  63. $user = $input['user'];
  64. $uid = $this->get_easemob_uid($user);//用户主键id
  65. //开始处理
  66. $loginToday = Db::name('user')->where('id',$uid)->whereTime('onlinetime', 'today')->find();
  67. if ($loginToday){
  68. \app\common\model\User::update(["is_online"=>1,"onlinetime"=>time()],["id"=>$uid]);
  69. }else{
  70. \app\common\model\User::update(["is_online"=>1,"onlinetime"=>time()],["id"=>$uid]);
  71. }
  72. exit;
  73. }
  74. //用户登出
  75. //用户登出(被其他设备踢掉)
  76. if(isset($input['reason']) && ( $input['reason'] == 'logout' || $input['status'] == 'replaced') && isset($input['status']) && $input['status'] == 'offline'){
  77. $user = $input['user'];
  78. $uid = $this->get_easemob_uid($user);//用户主键id
  79. //开始处理
  80. // 更新用户在线状态为离线
  81. \app\common\model\User::update(["is_online"=>0],["id"=>$uid]);
  82. \app\common\model\User::update(["is_live"=>0],["id"=>$uid]);
  83. $livingUserPartyId = $this->redis->hGet("livingUser", $uid);
  84. if ($livingUserPartyId) {
  85. // 扣除在线用户在房间情况
  86. $this->redis->hDel("livingUser", $uid);
  87. $this->redis->HDel("online_" . $livingUserPartyId, $uid);
  88. $this->redis->zRem("party_user_".$livingUserPartyId,$uid); //新加的
  89. }
  90. // 获取用户所在群信息
  91. $groupids = $this->getJoindGroupList($uid);
  92. // \app\common\model\Test::update(["content"=>json_encode($groupids)],["id"=>1]);
  93. if($groupids["ErrorCode"] == 0) {
  94. $grouplist = $groupids["GroupIdList"];
  95. $groupsites = [];
  96. if($grouplist) foreach($grouplist as $k => $v) {
  97. // 获取群组消息
  98. $groupInfo = $this->getGroupInfo($v["GroupId"]);
  99. if($groupInfo["ErrorCode"] == 0) $groupsites[] = $groupInfo["GroupInfo"][0];
  100. // 踢出用户在线组
  101. $this->redis->HDel("online_".$v["GroupId"],$uid);
  102. $this->redis->zRem("party_user_".$v["GroupId"],$uid); //新加的
  103. // 更新麦位前四
  104. $this->updatePosition($uid,$v["GroupId"]);
  105. // 取消排麦
  106. $this->cancelLineup($uid,$v["GroupId"]);
  107. // 房间状态变更
  108. $partyInfo = $this->redis->get("party_".$v["GroupId"]);
  109. if($partyInfo) {
  110. $partyInfo = json_decode($partyInfo,true);
  111. $memCount = count($this->redis->hGetAll("online_".$v["GroupId"]));
  112. if($memCount <= 0) {
  113. $partyInfo["is_online"] = 0;
  114. $this->redis->set("party_".$v["GroupId"],json_encode($partyInfo));
  115. \app\common\model\Party::update(["is_online"=>0],["id"=>$v["GroupId"]]);
  116. }
  117. }
  118. $liveInfo = $this->redis->get("live_".$v["GroupId"]);
  119. if($liveInfo) {
  120. $liveInfo = json_decode($liveInfo,true);
  121. $memCount = count($this->redis->hGetAll("online_".$v["GroupId"]));
  122. if($memCount <= 0) {
  123. $liveInfo["is_online"] = 0;
  124. $this->redis->set("live_".$v["GroupId"],json_encode($liveInfo));
  125. \app\common\model\Party::update(["is_online"=>0],["id"=>$v["GroupId"]]);
  126. }
  127. }
  128. }
  129. // 强制下麦
  130. print_r($this->downSite($uid,$groupsites));
  131. }
  132. exit;
  133. }
  134. //群组和聊天室操作:
  135. //muc:kick 踢出聊天室
  136. //muc:absence 有成员离开了聊天室
  137. //muc:leave 成员主动退出聊天室
  138. if(isset($input['chat_type']) && $input['chat_type'] == 'muc' && isset($input['eventType']) && $input['eventType'] == 'chat'){
  139. $room_id = '';
  140. $is_chatroom = false;
  141. $operation = false;
  142. $error_code = false;
  143. if(isset($input['payload'])){
  144. $payload = $input['payload'];
  145. if(isset($payload['muc_id'])){
  146. $room_id = $this->get_easemob_uid($payload['muc_id']);
  147. }
  148. if(isset($payload['is_chatroom']) && $payload['is_chatroom'] == true){
  149. $is_chatroom = true;
  150. }
  151. if(isset($payload['operation']) && in_array($payload['operation'],['kick','absence','leave']) ){
  152. $operation = true;
  153. }
  154. if(isset($payload['status']['error_code']) && $payload['status']['error_code'] == 'ok'){
  155. $error_code = true;
  156. }
  157. }
  158. //用户
  159. $uid = 0;
  160. if(isset($input['from'])){
  161. $uid = $this->get_easemob_uid($input['from']);
  162. }
  163. if($uid && $room_id && $is_chatroom && $operation && $error_code){
  164. //开始处理
  165. }
  166. }
  167. }
  168. /**
  169. * 获取用户所加入的聊天室
  170. */
  171. public function getJoindGroupList($user_id) {
  172. $random = rand(10000000,99999999);
  173. $usersig = $this->usersig("administrator");
  174. // 获取配置信息
  175. $config = config("tencent_im");
  176. $url = "https://console.tim.qq.com/v4/group_open_http_svc/get_joined_group_list";
  177. $url .= "?sdkappid=".$config["sdkappid"];
  178. $url .= "&identifier=administrator";
  179. $url .= "&usersig=".$usersig;
  180. $url .= "&random=".$random;
  181. $url .= "&contenttype=json";
  182. $tencentObj = new tencentim($url);
  183. $data = [];
  184. $data["Member_Account"] = $user_id;
  185. $data["WithHugeGroups"] = 1;
  186. $data["GroupType"] = 'AVChatRoom';
  187. $data["ResponseFilter"] = ["GroupBaseInfoFilter" => ['GroupId'] ];
  188. $groupInfo = $tencentObj->toSend($data);
  189. // \app\common\model\Test::update(["content"=>json_encode($groupInfo)],["id"=>1]);
  190. return $groupInfo;
  191. }
  192. /**
  193. * 获取群组信息
  194. */
  195. public function getGroupInfo($party_id='0') {
  196. $party_id = !empty($party_id) ? $party_id : $this->request->param('part_id',0);
  197. $random = rand(10000000,99999999);
  198. $usersig = $this->usersig("administrator");
  199. // 获取配置信息
  200. $config = config("tencent_im");
  201. $url = "https://console.tim.qq.com/v4/group_open_http_svc/get_group_info";
  202. $url .= "?sdkappid=".$config["sdkappid"];
  203. $url .= "&identifier=administrator";
  204. $url .= "&usersig=".$usersig;
  205. $url .= "&random==".$random;
  206. $url .= "&contenttype=json";
  207. $tencentObj = new tencentim($url);
  208. $data = [];
  209. $data["GroupIdList"] = [(string)$party_id];
  210. $data["ResponseFilter"] = [
  211. "GroupBaseInfoFilter" => ["GroupId","Type","Name","FaceUrl","Owner_Account","MemberNum"],
  212. "MemberInfoFilter" => ["Account","Role"],
  213. "AppDefinedDataFilter_Group" => ["roomInfo"],
  214. ];
  215. $groupInfo = $tencentObj->toSend($data);
  216. $this->success('获取成功',$groupInfo['GroupInfo']);
  217. }
  218. /**
  219. * 更新麦位前四
  220. */
  221. public function updatePosition($user_id,$party_id) {
  222. // 获取用户头像
  223. $userInfo = \app\common\model\User::field("avatar")->where(["id"=>$user_id])->find();
  224. $userAvatar = isset($userInfo["avatar"])?$userInfo["avatar"]:"";
  225. // 更新下麦时间
  226. $update = [];
  227. $update["offsite_time"] = time();
  228. $update["status"] = 2;
  229. \app\common\model\UserOnsiteTime::update($update,["user_id"=>$user_id,"status"=>1]);
  230. //
  231. $redis = new Redis();
  232. $redisconfig = config("redis");
  233. $redis->connect($redisconfig["host"], $redisconfig["port"]);
  234. if ($redisconfig['redis_pwd']) {
  235. $redis->auth($redisconfig['redis_pwd']);
  236. }
  237. if($redisconfig['redis_selectdb'] > 0){
  238. $redis->select($redisconfig['redis_selectdb']);
  239. }
  240. $room_type = Db::name('party')->where('id',$party_id)->value('room_type');
  241. $redisData = $redis->get($room_type."_".$party_id);
  242. if(!$redisData){
  243. return true;
  244. }
  245. $partyInfo = json_decode($redisData,true);
  246. // 遍历已有头像
  247. $partyuser = isset($partyInfo["party_user"])?$partyInfo["party_user"]:"";
  248. if(is_array($partyuser)) foreach($partyuser as $k => $v) if($v === $userAvatar) unset($partyInfo["party_user"][$k]);
  249. $redis->set($room_type."_".$party_id,json_encode($partyInfo));
  250. }
  251. /**
  252. * 取消排麦
  253. */
  254. public function cancelLineUp($user_id,$party_id) {
  255. if (!$party_id || !$user_id) {
  256. return false;
  257. }
  258. $redis = new Redis();
  259. $redisconfig = config("redis");
  260. $redis->connect($redisconfig["host"], $redisconfig["port"]);
  261. if ($redisconfig['redis_pwd']) {
  262. $redis->auth($redisconfig['redis_pwd']);
  263. }
  264. if($redisconfig['redis_selectdb'] > 0){
  265. $redis->select($redisconfig['redis_selectdb']);
  266. }
  267. $data=unserialize($redis->hGet("party_lineup",$party_id));
  268. // 更改红点排麦数量
  269. $num = 0;
  270. $send = false;
  271. if($data) foreach($data as $k => $v) {
  272. if($v["user_id"] == $user_id) {
  273. $send = true;
  274. unset($data[$k]);
  275. break;
  276. }
  277. }
  278. $redis->hSet("party_lineup",$party_id,serialize($data));
  279. if($send) {
  280. is_array($data) && $num = count($data);
  281. $random = rand(10000000,99999999);
  282. $usersig = $this->usersig("administrator");
  283. // 获取配置信息
  284. $config = config("tencent_im");
  285. $url = "https://console.tim.qq.com/v4/group_open_http_svc/send_group_msg";
  286. $url .= "?sdkappid=".$config["sdkappid"];
  287. $url .= "&identifier=administrator";
  288. $url .= "&usersig=".$usersig;
  289. $url .= "&random=".$random;
  290. $url .= "&contenttype=json";
  291. $tencentObj = new tencentim($url);
  292. $data = [];
  293. $data["GroupId"] = $party_id;
  294. $data["Random"] = rand(1000000,9999999);
  295. $message = [
  296. "message" => json_encode(["type"=>23,"content"=>$num]),
  297. "version" => "1.0",
  298. "action" => 301,
  299. "command" => ""
  300. ];
  301. $data["MsgBody"][] = [
  302. "MsgType" => "TIMCustomElem",
  303. "MsgContent" => [
  304. "Data"=> json_encode($message)
  305. ],
  306. ];
  307. $tencentObj->toSend($data);
  308. }
  309. }
  310. /**
  311. * 强制下线
  312. */
  313. public function downSite($user_id,$groupsites) {
  314. $random = rand(10000000,99999999);
  315. $usersig = $this->usersig("administrator");
  316. // 获取配置信息
  317. $config = config("tencent_im");
  318. $url = "https://console.tim.qq.com/v4/group_open_http_svc/modify_group_base_info";
  319. $url .= "?sdkappid=".$config["sdkappid"];
  320. $url .= "&identifier=administrator";
  321. $url .= "&usersig=".$usersig;
  322. $url .= "&random=".$random;
  323. $url .= "&contenttype=json";
  324. $tencentObj = new tencentim($url);
  325. // 先更新下麦时间
  326. $update = [];
  327. $update["offsite_time"] = time();
  328. $update["status"] = 2;
  329. \app\common\model\UserOnsiteTime::update($update,["user_id"=>$user_id,"status"=>1]);
  330. // 循环房间
  331. if($groupsites) foreach($groupsites as $k => $v) {
  332. // 循环座位
  333. $data = [];
  334. foreach($v["AppDefinedData"] as $m => $n) {
  335. // 解析字段信息
  336. $siteInfo = json_decode($n["Value"],true);
  337. if($m != 0 && isset($siteInfo["status"]) && $siteInfo["status"] == 1 && isset($siteInfo["status"]) && $siteInfo["user"] == $user_id) {
  338. $data["AppDefinedData"][] = [
  339. "Key" => $n["Key"],
  340. "Value" => '{"mute":false,"status":0,"user":""}',
  341. ];
  342. }
  343. }
  344. if(!empty($data)) {
  345. $data["GroupId"] = $v["GroupId"];
  346. return $tencentObj->toSend($data);
  347. }
  348. }
  349. }
  350. //输入: 1101231101159883#demo_9@easemob.com/android_f9f51032-3369-4f02-95bb-0cbf41b26837
  351. //输出: 9
  352. //输入: 1101231101159883#demo_231181574537218@admin.conference.easemob.com
  353. //输出: 231181574537218
  354. private function get_easemob_uid($user = ''){
  355. //去掉后半段
  356. $easemob = '@';
  357. $start = strpos($user,$easemob);
  358. $uid = substr($user,0,$start);
  359. //echo $uid;
  360. //echo '<br>';
  361. //去掉前缀
  362. $uid = substr($uid,strlen($this->appKey)+1);
  363. //echo $uid;
  364. return intval($uid);
  365. }
  366. //异步日志
  367. private function notify_log_start($paytype = 'easemob'){
  368. //记录支付回调数据
  369. ignore_user_abort(); // run script in background
  370. set_time_limit(30);
  371. // 日志文件 start
  372. $log_base_dir = '../paylog/'.$paytype.'/';
  373. if (!is_dir($log_base_dir))
  374. {
  375. mkdir($log_base_dir, 0770, true);
  376. @chmod($log_base_dir, 0770);
  377. }
  378. $notify_file = $log_base_dir.'notify.txt';
  379. if(!file_exists($notify_file)) {
  380. @touch($notify_file);
  381. @chmod($notify_file, 0770);
  382. }
  383. if(filesize($notify_file)>5242880)//大于5M自动切换
  384. {
  385. rename($notify_file, $log_base_dir.'notify_'.date('Y_m_d_H_i_s').'.txt');
  386. }
  387. if(!file_exists($notify_file)) {
  388. @touch($notify_file);
  389. @chmod($notify_file, 0770);
  390. }
  391. // 日志文件 end
  392. //开始写入
  393. $xml = file_get_contents("php://input");
  394. file_put_contents($notify_file, "\r\n\r\n".date('Y-m-d H:i:s')." [notify][入口接收php://input流原始数据] \n".$xml, FILE_APPEND);
  395. ini_set('display_errors','On');
  396. return $notify_file;
  397. }
  398. }