Easemob.php 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503
  1. <?php
  2. namespace app\api\controller;
  3. use app\common\controller\Api;
  4. use Redis;
  5. use think\Db;
  6. use app\common\library\Easemob as Easemoblib;
  7. /**
  8. * 环信接口
  9. */
  10. class Easemob extends Api
  11. {
  12. protected $noNeedLogin = ["callback"];
  13. protected $noNeedRight = ['*'];
  14. protected $appKey;
  15. protected $redis;
  16. public function __construct() {
  17. parent::__construct();
  18. //初始化配置
  19. $easemob_config = config('easemob');
  20. $this->appKey = $easemob_config['appkey'];
  21. //redis
  22. $redis = new Redis();
  23. $redisconfig = config("redis");
  24. $redis->connect($redisconfig["host"], $redisconfig["port"]);
  25. if ($redisconfig['redis_pwd']) {
  26. $redis->auth($redisconfig['redis_pwd']);
  27. }
  28. if($redisconfig['redis_selectdb'] > 0){
  29. $redis->select($redisconfig['redis_selectdb']);
  30. }
  31. $this->redis = $redis;
  32. }
  33. /**
  34. * 回调
  35. * https://docs-im-beta.easemob.com/document/server-side/callback_configurations.html
  36. *
  37. * 群组和聊天室操作:
  38. *
  39. * muc:presence 有新成员加入了聊天室
  40. *{"callId":"1101231101159883#demo_1215090885268604500","channel_channel":"1101231101159883#demo_230909515202562@admin.conference.easemob.com","eventType":"chat","channel_user":"1101231101159883#demo_9@easemob.com","chat_type":"muc","security":"845b22a4edcb4fde2185ff7f0374f051","is_downgrade":false,"content_type":"muc:presence","payload":{"muc_id":"1101231101159883#demo_230909515202562@conference.easemob.com","is_chatroom":true,"operation":"presence"},"group_id":"230909515202562","writed_channel":false,"host":"msync@ebs-ali-beijing-msync105","appkey":"1101231101159883#demo","from":"1101231101159883#demo_9@easemob.com","to":"230909515202562","msg_id":"1215090885268604500","timestamp":1700475192937}
  41. *
  42. * muc:kick 踢出聊天室
  43. * {"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}
  44. *
  45. * muc:absence 有成员离开了聊天室
  46. * {"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}
  47. *
  48. * muc:leave 成员主动退出聊天室
  49. * {"callId":"1101231101159883#demo_1216580964256318924","channel_channel":"1101231101159883#demo_231346001739777@conference.easemob.com","eventType":"chat","channel_user":"1101231101159883#demo_231346001739777@admin.conference.easemob.com","chat_type":"muc","security":"059f0313db99c98cde55b42d28538f9a","is_downgrade":false,"content_type":"muc:leave","payload":{"muc_id":"1101231101159883#demo_231346001739777@conference.easemob.com","is_chatroom":true,"operation":"leave","status":{"description":"","error_code":"ok"}},"group_id":"231346001739777","writed_channel":false,"host":"msync@ebs-ali-beijing-msync71","appkey":"1101231101159883#demo","from":"1101231101159883#demo_9@easemob.com","to":"231346001739777","msg_id":"1216580964256318924","timestamp":1700822128988}
  50. *
  51. * 用户登入
  52. * {"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"}
  53. *
  54. * 用户登出
  55. * {"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"}
  56. *
  57. * 用户登出(被其他设备踢掉)
  58. * {"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"}
  59. */
  60. //登入,登出,进入房间其实都不需要处理。登入不影响房间。只登出只要不影响房间,在退出房间时间会重复回调。登入房间有接口,这里没必要重复处理。就只有退出房间有用也简单
  61. public function callback() {
  62. $this->notify_log_start();
  63. $input = file_get_contents("php://input"); // 主题信息
  64. $input = json_decode($input,true);
  65. //验证加密
  66. //解析数据格式
  67. //用户登入
  68. if(isset($input['reason']) && $input['reason'] == 'login' && isset($input['status']) && $input['status'] == 'online'){
  69. $user = $input['user'];
  70. $uid = $this->get_easemob_uid($user);//用户主键id
  71. //开始处理
  72. $loginToday = Db::name('user')->where('id',$uid)->whereTime('onlinetime', 'today')->find();
  73. if ($loginToday){
  74. \app\common\model\User::update(["is_online"=>1,"onlinetime"=>time()],["id"=>$uid]);
  75. }else{
  76. \app\common\model\User::update(["is_online"=>1,"onlinetime"=>time()],["id"=>$uid]);
  77. }
  78. exit;
  79. }
  80. //用户登出
  81. //用户登出(被其他设备踢掉)
  82. if(isset($input['reason']) && ( $input['reason'] == 'logout' || $input['status'] == 'replaced') && isset($input['status']) && $input['status'] == 'offline'){
  83. $user = $input['user'];
  84. $uid = $this->get_easemob_uid($user);//用户主键id
  85. //开始处理
  86. // 更新用户在线状态为离线
  87. \app\common\model\User::update(["is_online"=>0],["id"=>$uid]);
  88. \app\common\model\User::update(["is_live"=>0],["id"=>$uid]);
  89. $livingUserPartyId = $this->redis->hGet("livingUser", $uid);
  90. if ($livingUserPartyId) {
  91. // 扣除在线用户在房间情况
  92. $this->redis->hDel("livingUser", $uid);
  93. $this->redis->HDel("online_" . $livingUserPartyId, $uid);
  94. $this->redis->zRem("party_user_".$livingUserPartyId,$uid); //新加的
  95. }
  96. // 获取用户所在群信息
  97. $grouplist = $this->getJoindGroupList($uid);
  98. if(!empty($grouplist)) {
  99. $groupsites = [];
  100. if($grouplist) foreach($grouplist as $k => $v) {
  101. $partyid = Db::name('party')->where('easemob_room_id',$v['id'])->value('id');
  102. if(empty($partyid)){
  103. continue;
  104. }
  105. $v["GroupId"] = $partyid;
  106. // 获取群组消息
  107. // $groupInfo = $this->getGroupInfo($v["GroupId"]); 因为downSite方法没用到,所以屏蔽
  108. // if($groupInfo["ErrorCode"] == 0) $groupsites[] = $groupInfo["GroupInfo"][0]; 因为downSite方法没用到,所以屏蔽
  109. // 踢出用户在线组
  110. $this->redis->HDel("online_".$v["GroupId"],$uid);
  111. $this->redis->zRem("party_user_".$v["GroupId"],$uid); //新加的
  112. // 更新麦位前四
  113. $this->updatePosition($uid,$v["GroupId"]);
  114. // 取消排麦
  115. //$this->cancelLineup($uid,$v["GroupId"]);
  116. // 房间状态变更
  117. $partyInfo = $this->redis->get("party_".$v["GroupId"]);
  118. if($partyInfo) {
  119. $partyInfo = json_decode($partyInfo,true);
  120. $memCount = count($this->redis->hGetAll("online_".$v["GroupId"]));
  121. if($memCount <= 0) {
  122. $partyInfo["is_online"] = 0;
  123. $this->redis->set("party_".$v["GroupId"],json_encode($partyInfo));
  124. \app\common\model\Party::update(["is_online"=>0],["id"=>$v["GroupId"]]);
  125. }
  126. }
  127. $liveInfo = $this->redis->get("live_".$v["GroupId"]);
  128. if($liveInfo) {
  129. $liveInfo = json_decode($liveInfo,true);
  130. $memCount = count($this->redis->hGetAll("online_".$v["GroupId"]));
  131. if($memCount <= 0) {
  132. $liveInfo["is_online"] = 0;
  133. $this->redis->set("live_".$v["GroupId"],json_encode($liveInfo));
  134. \app\common\model\Party::update(["is_online"=>0],["id"=>$v["GroupId"]]);
  135. }
  136. }
  137. }
  138. // 强制下麦
  139. print_r($this->downSite($uid,$groupsites));
  140. //理论上,应该轮询所有房间,轮询这个用户所在的麦位,给下麦,如果是房主,应该解散
  141. }
  142. exit;
  143. }
  144. //群组和聊天室操作:
  145. //muc:presence 有新成员加入了聊天室
  146. //muc:kick 踢出聊天室
  147. //muc:absence 有成员离开了聊天室
  148. //muc:leave 成员主动退出聊天室
  149. if(isset($input['chat_type']) && $input['chat_type'] == 'muc' && isset($input['eventType']) && $input['eventType'] == 'chat'){
  150. $room_id = '';
  151. $is_chatroom = false;
  152. $operation = false;
  153. $error_code = false;
  154. if(isset($input['payload'])){
  155. $payload = $input['payload'];
  156. if(isset($payload['muc_id'])){
  157. $room_id = $this->get_easemob_uid($payload['muc_id']);
  158. }
  159. if(isset($payload['is_chatroom']) && ($payload['is_chatroom'] == true || $payload['is_chatroom'] == 1) ){
  160. $is_chatroom = true;
  161. }
  162. if(isset($payload['operation']) && in_array($payload['operation'],['kick','absence','leave']) ){
  163. $operation = 'out';
  164. }
  165. if(isset($payload['operation']) && $payload['operation'] == 'presence' ){
  166. $operation = 'in';
  167. $operation = false;//进入房间走接口,不在这里处理
  168. }
  169. if(isset($payload['status']['error_code']) && $payload['status']['error_code'] == 'ok'){
  170. $error_code = true;
  171. }
  172. }
  173. //用户
  174. $uid = 0;
  175. if(isset($input['from'])){
  176. $uid = $this->get_easemob_uid($input['from']);
  177. }
  178. $partyid = Db::name('party')->where('easemob_room_id',$room_id)->value('id');
  179. //开始处理
  180. //进入房间
  181. if($uid && $partyid && $is_chatroom && $operation == 'in' && $error_code){
  182. $userId = $uid;
  183. $this->redis->hSet("online_" . $partyid, $userId, $userId);
  184. $this->redis->zAdd("party_user_".$partyid,$userId,$userId); //新加的
  185. // 记录在线用户在房间情况
  186. $this->redis->hSet("livingUser",$userId,$partyid);
  187. // 更新房间在线状态
  188. $partyInfo = $this->redis->get("party_" . $partyid);
  189. if ($partyInfo) {
  190. $partyInfo = json_decode($partyInfo, true);
  191. if ($partyInfo["is_online"] != 1) {
  192. $partyInfo["is_online"] = 1;
  193. $this->redis->set("party_" . $partyid, json_encode($partyInfo));
  194. \app\common\model\Party::update(["is_online" => 1], ["id" => $partyid]);
  195. }
  196. }
  197. // 更新房间在线状态
  198. $liveInfo = $this->redis->get("live_" . $partyid);
  199. if ($liveInfo) {
  200. $liveInfo = json_decode($liveInfo, true);
  201. if ($liveInfo["is_online"] != 1) {
  202. $liveInfo["is_online"] = 1;
  203. $this->redis->set("live_" . $partyid, json_encode($liveInfo));
  204. \app\common\model\Party::update(["is_online" => 1], ["id" => $partyid]);
  205. $rs_user = Db::name('user')->where('id',$userId)->update(['is_live'=>1]);
  206. }
  207. }
  208. //[环信] 更新聊天室自定义属性 在线用户人数
  209. $easemob = new Easemoblib();
  210. $matedata = [
  211. 'online_user_num' => count($this->redis->hGetAll("online_".$partyid)),
  212. ];
  213. $easemob->room_setRoomCustomAttributeForced($room_id,$uid,$matedata);
  214. }
  215. //出去房间
  216. if($uid && $partyid && $is_chatroom && $operation == 'out' && $error_code){
  217. $userId = $uid;
  218. $this->redis->HDel("online_" . $partyid, $userId);
  219. $this->redis->zRem("party_user_".$partyid,$userId); //新加的
  220. // 扣除在线用户在房间情况
  221. $this->redis->hDel("livingUser",$userId);
  222. // 更新房间在线状态
  223. $partyInfo = $this->redis->get("party_" . $partyid);
  224. if ($partyInfo) {
  225. $partyInfo = json_decode($partyInfo, true);
  226. $memCount = count($this->redis->hGetAll("online_" . $partyid));
  227. if ($memCount <= 0) {
  228. $partyInfo["is_online"] = 0;
  229. $this->redis->set("party_" . $partyid, json_encode($partyInfo));
  230. \app\common\model\Party::update(["is_online" => 0], ["id" => $partyid]);
  231. }
  232. }
  233. //
  234. $liveInfo = $this->redis->get("live_" . $partyid);
  235. if ($liveInfo) {
  236. $liveInfo = json_decode($liveInfo, true);
  237. $memCount = count($this->redis->hGetAll("online_" . $partyid));
  238. if ($memCount <= 0) {
  239. $liveInfo["is_online"] = 0;
  240. $this->redis->set("live_" . $partyid, json_encode($liveInfo));
  241. \app\common\model\Party::update(["is_online" => 0], ["id" => $partyid]);
  242. $rs_user = Db::name('user')->where('id',$userId)->update(['is_live'=>0]);
  243. }
  244. }
  245. //退出房间获取房间自定义属性
  246. //清理这个人的麦序
  247. //重置房间在线人数
  248. $this->clean_room($partyid,$uid);
  249. }
  250. exit;
  251. }
  252. }
  253. //清除某个麦位
  254. //重置房间在线人数
  255. private function clean_room($easemob_room_id,$uid){
  256. $party_info = Db::name('party')->field('id,user_id')->where('easemob_room_id',$easemob_room_id)->find();
  257. if(empty($party_info)){
  258. return false;
  259. }
  260. //获取所有麦位,假设8个
  261. $easemob = new Easemoblib();
  262. $seatlist = $easemob->room_getRoomCustomAttribute($easemob_room_id,['seat0','seat1','seat2','seat3','seat4','seat5','seat6','seat7']);
  263. if(empty($seatlist)){
  264. //默认为空
  265. return false;
  266. }
  267. $newseat = [
  268. 'charm' => 0, //红心,魅力值
  269. 'isMaster' => false, // 是否是房主
  270. 'headUrl' => '', // 头像
  271. 'userNo' => '', // 座位上用户no
  272. 'rtcUid' => '', // 座位上用户id,与rtc的userId一致
  273. 'name' => '', // 座位上用户昵称
  274. 'seatIndex' => -1, // 座位编号
  275. 'chorusSongCode' => '', // 是否合唱
  276. 'isAudioMuted' => 1, // 是否静音
  277. 'isVideoMuted' => 0, // 是否开启视频
  278. 'checked' => false, // 用于送礼物选择用户
  279. 'isUsed' => true, // 用于送礼物选择用户
  280. 'gender' => 1, //性别
  281. ];
  282. $matedata = [];
  283. foreach($seatlist as $k => $seat){
  284. $seat = json_decode($seat,true);
  285. //找到当前用户所在的麦位
  286. if(isset($seat['rtcUid']) && !empty($seat['rtcUid']) && $seat['rtcUid'] == $uid){
  287. //$newseat['seatIndex'] = substr($k,-1);
  288. //初始化这个麦位
  289. $matedata = [
  290. $k => json_encode($newseat),
  291. ];
  292. }
  293. }
  294. //重置房间在线人数
  295. $matedata['online_user_num'] = count($this->redis->hGetAll("online_".$party_info['id']));
  296. if(!empty($matedata)){
  297. $easemob->room_setRoomCustomAttributeForced($easemob_room_id,$party_info['user_id'],$matedata);
  298. }
  299. return true;
  300. }
  301. /**
  302. * 获取用户所加入的聊天室
  303. */
  304. private function getJoindGroupList($user_id) {
  305. $easemob = new Easemoblib();
  306. $list = $easemob->room_listAllRoomsUserJoined($user_id);
  307. return $list;
  308. }
  309. /**
  310. * 获取群组信息
  311. */
  312. private function getGroupInfo($easemob_room_id = '230909515202562') {
  313. $easemob = new Easemoblib();
  314. $info = $easemob->room_getRoom($easemob_room_id);
  315. return $info;
  316. }
  317. /**
  318. * 更新麦位前四
  319. */
  320. private function updatePosition($user_id,$party_id) {
  321. // 获取用户头像
  322. $userInfo = \app\common\model\User::field("avatar")->where(["id"=>$user_id])->find();
  323. $userAvatar = isset($userInfo["avatar"])?$userInfo["avatar"]:"";
  324. // 更新下麦时间
  325. $update = [];
  326. $update["offsite_time"] = time();
  327. $update["status"] = 2;
  328. \app\common\model\UserOnsiteTime::update($update,["user_id"=>$user_id,"status"=>1]);
  329. //
  330. $room_type = Db::name('party')->where('id',$party_id)->value('room_type');
  331. $redisData = $this->redis->get($room_type."_".$party_id);
  332. if(!$redisData){
  333. return true;
  334. }
  335. $partyInfo = json_decode($redisData,true);
  336. // 遍历已有头像
  337. $partyuser = isset($partyInfo["party_user"])?$partyInfo["party_user"]:"";
  338. if(is_array($partyuser)) foreach($partyuser as $k => $v) if($v === $userAvatar) unset($partyInfo["party_user"][$k]);
  339. $this->redis->set($room_type."_".$party_id,json_encode($partyInfo));
  340. }
  341. /**
  342. * 取消排麦
  343. */
  344. private function cancelLineUp($user_id,$party_id) {
  345. return false;
  346. //因为没有排麦了,设计图就没这个功能,所以后面没了
  347. }
  348. /**
  349. * 强制下线
  350. */
  351. private function downSite($user_id,$groupsites) {
  352. /*此处删减参考ggyuyin,api/tenim/downsite*/
  353. // 先更新下麦时间
  354. $update = [];
  355. $update["offsite_time"] = time();
  356. $update["status"] = 2;
  357. \app\common\model\UserOnsiteTime::update($update,["user_id"=>$user_id,"status"=>1]);
  358. // 循环房间
  359. /*此处删减参考ggyuyin,api/tenim/downsite*/
  360. }
  361. //输入: 1101231101159883#demo_9@easemob.com/android_f9f51032-3369-4f02-95bb-0cbf41b26837
  362. //输出: 9
  363. //输入: 1101231101159883#demo_231181574537218@admin.conference.easemob.com
  364. //输出: 231181574537218
  365. private function get_easemob_uid($user = ''){
  366. //去掉后半段
  367. $easemob = '@';
  368. $start = strpos($user,$easemob);
  369. $uid = substr($user,0,$start);
  370. //echo $uid;
  371. //echo '<br>';
  372. //去掉前缀
  373. $uid = substr($uid,strlen($this->appKey)+1);
  374. //echo $uid;
  375. return intval($uid);
  376. }
  377. //异步日志
  378. private function notify_log_start($paytype = 'easemob'){
  379. //记录支付回调数据
  380. ignore_user_abort(); // run script in background
  381. set_time_limit(30);
  382. // 日志文件 start
  383. $log_base_dir = '../paylog/'.$paytype.'/';
  384. if (!is_dir($log_base_dir))
  385. {
  386. mkdir($log_base_dir, 0770, true);
  387. @chmod($log_base_dir, 0770);
  388. }
  389. $notify_file = $log_base_dir.'notify.txt';
  390. if(!file_exists($notify_file)) {
  391. @touch($notify_file);
  392. @chmod($notify_file, 0770);
  393. }
  394. if(filesize($notify_file)>5242880)//大于5M自动切换
  395. {
  396. rename($notify_file, $log_base_dir.'notify_'.date('Y_m_d_H_i_s').'.txt');
  397. }
  398. if(!file_exists($notify_file)) {
  399. @touch($notify_file);
  400. @chmod($notify_file, 0770);
  401. }
  402. // 日志文件 end
  403. //开始写入
  404. $xml = file_get_contents("php://input");
  405. file_put_contents($notify_file, "\r\n\r\n".date('Y-m-d H:i:s')." [notify][入口接收php://input流原始数据] \n".$xml, FILE_APPEND);
  406. ini_set('display_errors','On');
  407. return $notify_file;
  408. }
  409. }