RoomServiceImpl.java 83 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677
  1. package com.ym.service.Impl;
  2. import com.alibaba.fastjson.JSON;
  3. import com.alibaba.fastjson.JSONObject;
  4. import com.alibaba.fastjson.serializer.SerializerFeature;
  5. import com.ym.common.ApiException;
  6. import com.ym.common.BaseResponse;
  7. import com.ym.common.DisplayEnum;
  8. import com.ym.common.ErrorEnum;
  9. import com.ym.config.IMProperties;
  10. import com.ym.config.RoomProperties;
  11. import com.ym.dao.*;
  12. import com.ym.job.ScheduleManager;
  13. import com.ym.mec.auth.api.client.SysUserFeignService;
  14. import com.ym.mec.auth.api.entity.SysUser;
  15. import com.ym.mec.biz.dal.dao.*;
  16. import com.ym.mec.biz.dal.dao.CourseScheduleStudentMusicScoreDao;
  17. import com.ym.mec.biz.dal.dto.BasicUserDto;
  18. import com.ym.mec.biz.dal.dto.RongyunBasicUserDto;
  19. import com.ym.mec.biz.dal.entity.*;
  20. import com.ym.mec.biz.dal.enums.GroupType;
  21. import com.ym.mec.biz.dal.enums.TeachModeEnum;
  22. import com.ym.mec.biz.service.StudentAttendanceService;
  23. import com.ym.mec.biz.service.SysConfigService;
  24. import com.ym.mec.biz.service.SysTenantConfigService;
  25. import com.ym.mec.biz.service.TeacherAttendanceService;
  26. import com.ym.mec.common.exception.BizException;
  27. import com.ym.mec.im.IMHelper;
  28. import com.ym.mec.im.message.*;
  29. import com.ym.mec.util.collection.MapUtil;
  30. import com.ym.mec.util.date.DateUtil;
  31. import com.ym.pojo.*;
  32. import com.ym.service.RoomService;
  33. import com.ym.utils.CheckUtils;
  34. import com.ym.utils.CodeUtil;
  35. import com.ym.utils.DateTimeUtils;
  36. import com.ym.utils.IdentifierUtils;
  37. import com.ym.whiteboard.WhiteBoardHelper;
  38. import lombok.extern.slf4j.Slf4j;
  39. import org.apache.commons.lang3.StringUtils;
  40. import org.springframework.beans.factory.annotation.Autowired;
  41. import org.springframework.data.redis.core.RedisTemplate;
  42. import org.springframework.stereotype.Service;
  43. import org.springframework.transaction.annotation.Isolation;
  44. import org.springframework.transaction.annotation.Propagation;
  45. import org.springframework.transaction.annotation.Transactional;
  46. import java.util.*;
  47. import java.util.concurrent.TimeUnit;
  48. import java.util.stream.Collectors;
  49. import static com.ym.pojo.RoleEnum.RoleStudent;
  50. import static com.ym.pojo.RoleEnum.RoleTeacher;
  51. /**
  52. * Created by super_zou on 2019/11/28.
  53. */
  54. @Slf4j
  55. @Service
  56. public class RoomServiceImpl implements RoomService {
  57. @Autowired
  58. private IMHelper imHelper;
  59. @Autowired
  60. private RoomProperties roomProperties;
  61. @Autowired
  62. private RoomDao roomDao;
  63. @Autowired
  64. private RoomMemberDao roomMemberDao;
  65. @Autowired
  66. private CourseScheduleStudentPaymentDao courseScheduleStudentPaymentDao;
  67. @Autowired
  68. private WhiteBoardHelper whiteBoardHelper;
  69. @Autowired
  70. private WhiteboardDao whiteboardDao;
  71. @Autowired
  72. private ScheduleManager scheduleManager;
  73. @Autowired
  74. private UserDao userDao;
  75. @Autowired
  76. private TeacherDao teacherDao;
  77. @Autowired
  78. private CourseScheduleDao courseScheduleDao;
  79. @Autowired
  80. private TeacherAttendanceService teacherAttendanceService;
  81. @Autowired
  82. private StudentAttendanceService studentAttendanceService;
  83. @Autowired
  84. private IMProperties imProperties;
  85. @Autowired
  86. private SysUserFeignService sysUserFeignService;
  87. @Autowired
  88. private SysExamSongDao sysExamSongDao;
  89. @Autowired
  90. private SysConfigDao sysConfigDao;
  91. @Autowired
  92. private CourseScheduleStudentMusicScoreDao courseScheduleStudentMusicScoreDao;
  93. @Autowired
  94. private SysMusicScoreAccompanimentDao sysMusicScoreAccompanimentDao;
  95. @Autowired
  96. private SysTenantConfigService sysTenantConfigService;
  97. @Autowired
  98. private RedisTemplate<String,String> redisTemplate;
  99. @Override
  100. public Integer getCurrentCourseId(String roomId){
  101. CourseSchedule courseSchedule = courseScheduleDao.get(Long.parseLong(roomId));
  102. //是否是连堂课
  103. String continueCourseTime = sysTenantConfigService.getTenantConfigValue(SysConfigService.ONLINE_CONTINUE_COURSE_TIME,courseSchedule.getTenantId());
  104. if (StringUtils.isEmpty(continueCourseTime)) {
  105. continueCourseTime = "5";
  106. }
  107. CourseSchedule schedule = courseSchedule;
  108. //如果当前课程是连堂课,那么获取第一节课的课程编号
  109. while (true){
  110. String classDate = DateUtil.format(schedule.getClassDate(), DateUtil.DEFAULT_PATTERN);
  111. String startClassTime = DateUtil.format(schedule.getStartClassTime(), DateUtil.EXPANDED_TIME_FORMAT);
  112. schedule = courseScheduleDao.getLastCourse(schedule.getClassGroupId(),classDate + " " + startClassTime,schedule.getActualTeacherId(),continueCourseTime);
  113. if(schedule != null){
  114. roomId = schedule.getId().toString();
  115. }else {
  116. break;
  117. }
  118. }
  119. return Integer.parseInt(roomId);
  120. }
  121. public static void main(String[] args) {
  122. Date date = DateUtil.stringToDate("2021-12-14 11:34:00", DateUtil.EXPANDED_DATE_TIME_FORMAT);
  123. Date classDate = DateUtil.stringToDate("2021-12-14 11:40:00", DateUtil.EXPANDED_DATE_TIME_FORMAT);
  124. int i = classDate.compareTo(DateUtil.addMinutes(date, 5));
  125. int i1 = DateUtil.minutesBetween(classDate, date);
  126. }
  127. @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
  128. @Override
  129. public RoomResult joinRoom(String roomId) throws Exception {
  130. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  131. SysUser sysUser = sysUserFeignService.queryUserInfo();
  132. String userId = sysUser.getId().toString();
  133. log.info("joinRoom: roomId={}, userId={}", roomId, userId);
  134. Teacher teacher = teacherDao.get(Integer.parseInt(userId));
  135. CourseSchedule courseSchedule = courseScheduleDao.get(Long.parseLong(roomId));
  136. if(courseSchedule.getTeachMode() == TeachModeEnum.OFFLINE){
  137. throw new ApiException(ErrorEnum.JOIN_ROOM_ERROR);
  138. }
  139. Date curTime = DateTimeUtils.currentUTC();
  140. //是否提前进入教室
  141. String courseBeforeBufferTime = sysTenantConfigService.getTenantConfigValue(SysConfigService.COURSE_BEFORE_BUFFER_TIME,courseSchedule.getTenantId());
  142. if (StringUtils.isEmpty(courseBeforeBufferTime)) {
  143. courseBeforeBufferTime = "5";
  144. }
  145. Date addMinutes = DateUtil.addMinutes(curTime, Integer.parseInt(courseBeforeBufferTime));
  146. if(courseSchedule.getStartClassTime().compareTo(addMinutes) > 0 ){
  147. throw new ApiException(ErrorEnum.ROOM_NOT_START);
  148. // throw new BizException("网络教室暂未开启,请在{}分钟后进入教室",DateUtil.minutesBetween(addMinutes,courseSchedule.getStartClassTime()));
  149. }
  150. //是否是连堂课
  151. String continueCourseTime = sysTenantConfigService.getTenantConfigValue(SysConfigService.ONLINE_CONTINUE_COURSE_TIME,courseSchedule.getTenantId());
  152. if (StringUtils.isEmpty(continueCourseTime)) {
  153. continueCourseTime = "5";
  154. }
  155. String autoCloseNetworkRoomTime = sysTenantConfigService.getTenantConfigValue(SysConfigService.COURSE_BEFORE_BUFFER_TIME,courseSchedule.getTenantId());
  156. if(StringUtils.isEmpty(autoCloseNetworkRoomTime)){
  157. autoCloseNetworkRoomTime = "15";
  158. }
  159. RoomResult roomResult = new RoomResult();
  160. roomResult.setAutoCloseNetworkRoomTime(autoCloseNetworkRoomTime);
  161. CourseSchedule schedule = courseSchedule;
  162. //如果当前课程是连堂课,那么获取第一节课的课程编号
  163. while (true){
  164. String classDate = DateUtil.format(schedule.getClassDate(), DateUtil.DEFAULT_PATTERN);
  165. String startClassTime = DateUtil.format(schedule.getStartClassTime(), DateUtil.EXPANDED_TIME_FORMAT);
  166. schedule = courseScheduleDao.getLastCourse(schedule.getClassGroupId(),classDate + " " + startClassTime,schedule.getActualTeacherId(),continueCourseTime);
  167. if(schedule != null){
  168. roomId = schedule.getId().toString();
  169. roomResult.setAutoCloseFlag(false);
  170. }else {
  171. break;
  172. }
  173. }
  174. Long courseId = Long.parseLong(roomId);
  175. //记录用户实际选择的房间
  176. if(courseSchedule.getGroupType() == GroupType.COMM){
  177. roomId = "I" + roomId;
  178. }else {
  179. roomId = "S" + roomId;
  180. }
  181. redisTemplate.opsForValue().setIfAbsent(roomId + userId,courseSchedule.getId().toString());
  182. log.info("joinRoom current: roomId={}, userId={}", roomId, userId);
  183. RoleEnum roleEnum;
  184. RoomResult.MemberResult userResult = new RoomResult.MemberResult();
  185. RoomMember member = roomMemberDao.findByRidAndUid(roomId, userId);
  186. String userName;
  187. if (member == null) {
  188. int count = roomMemberDao.countByRidAndExcludeRole(roomId, RoleEnum.RoleAudience.getValue());
  189. if (count == roomProperties.getMaxCount()) {
  190. log.info("join error Over max count: roomId = {}, userId = {}", roomId,userId);
  191. throw new ApiException(ErrorEnum.ERR_OVER_MAX_COUNT);
  192. }
  193. if(teacher != null && teacher.getId().equals(courseSchedule.getActualTeacherId())){
  194. roleEnum = RoleTeacher;
  195. userName = sysUser.getRealName();
  196. }else {
  197. roleEnum = RoleStudent;
  198. userName = sysUser.getUsername();
  199. }
  200. userResult.setMicrophone(true);
  201. userResult.setCamera(true);
  202. userResult.setHandUpOn(false);
  203. userResult.setJoinTime(curTime);
  204. saveRoomMember(userId,sysUser.getAvatar(), userName, roomId, roleEnum.getValue(), curTime);
  205. } else {
  206. roleEnum = RoleEnum.getEnumByValue(member.getRole());
  207. if(roleEnum == RoleTeacher){
  208. courseScheduleStudentPaymentDao.adjustPlayMidi(courseId,null,null);
  209. }
  210. userName = member.getName();
  211. userResult.setCamera(member.isCamera());
  212. userResult.setHandUpOn(member.isHand());
  213. userResult.setJoinTime(member.getJoinDt());
  214. }
  215. imHelper.joinGroup(new String[]{userId}, roomId, roomId);
  216. List<CourseScheduleStudentMusicScore> scheduleStudentMusicScores = courseScheduleStudentMusicScoreDao.queryByScoreIdAndCourseId(null, courseId,null, null, null);
  217. Room room = roomDao.findByRid(roomId);
  218. String display = "";
  219. if (roleEnum == RoleTeacher) {
  220. display = "display://type=1?userId=" + userId + "?uri=";
  221. } else if (roleEnum == RoleEnum.RoleAssistant && display.isEmpty()) {
  222. display = "display://type=0?userId=" + userId + "?uri=";
  223. }else {
  224. ExamSongDownloadData examSongDownloadData;
  225. String json = courseScheduleStudentPaymentDao.getExamJsonByCourseIdAndUserId(courseId, sysUser.getId());
  226. if(StringUtils.isEmpty(json)){
  227. examSongDownloadData = new ExamSongDownloadData();
  228. }else {
  229. examSongDownloadData = JSON.parseObject(json,ExamSongDownloadData.class);
  230. }
  231. //获取学员曲目下载状态
  232. userResult.setExamSongDownloadJson(examSongDownloadData);
  233. if (room != null) {
  234. display = room.getDisplay();
  235. }
  236. }
  237. //已下载的伴奏列表
  238. if(scheduleStudentMusicScores != null && scheduleStudentMusicScores.size() > 0){
  239. List<CourseScheduleStudentMusicScore> musicScores = scheduleStudentMusicScores.stream().filter(e -> e.getUserId().equals(sysUser.getId())).collect(Collectors.toList());
  240. String toJSONString = JSON.toJSONString(musicScores, SerializerFeature.DisableCircularReferenceDetect);
  241. List<CourseScheduleStudentMusicScore> lists = JSON.parseArray(toJSONString, CourseScheduleStudentMusicScore.class);
  242. userResult.setScheduleStudentMusicScores(lists);
  243. }
  244. userResult.setUserName(userName);
  245. userResult.setUserId(userId);
  246. userResult.setRole(roleEnum.getValue());
  247. //获取节拍器信息
  248. String midi = courseScheduleStudentPaymentDao.getMidiByCourseIdAndUserId(courseId.toString(),userId);
  249. userResult.setPlayMidiJson(JSONObject.parseObject(midi,CustomMessage.class));
  250. //获取当前课程剩余时长
  251. // String classDate = DateUtil.format(courseSchedule.getClassDate(), DateUtil.DEFAULT_PATTERN);
  252. // String endClassTime = DateUtil.format(courseSchedule.getEndClassTime(), DateUtil.EXPANDED_TIME_FORMAT);
  253. roomResult.setSurplusTime(DateUtil.secondsBetween(new Date(),courseSchedule.getEndClassTime()));
  254. roomResult.setUserInfo(userResult);
  255. roomResult.setDisplay(display);
  256. roomResult.setRoomId(roomId);
  257. List<RoomMember> roomMemberList = roomMemberDao.findByRid(roomId);
  258. if(roomMemberList != null && roomMemberList.size() > 0){
  259. Set<String> userIds = roomMemberList.stream().map(e -> e.getUid()).collect(Collectors.toSet());
  260. Map<Integer,String> midiMap = MapUtil.convertMybatisMap(courseScheduleStudentPaymentDao.queryMidiByUserIdsAndCourseId(userIds,courseId.toString()));
  261. Map<Integer,String> examSongMap = MapUtil.convertMybatisMap(courseScheduleStudentPaymentDao.queryExamSongByUserIdsAndCourseId(userIds,courseId.toString()));
  262. roomResult.setMembers(roomMemberList,midiMap,examSongMap,scheduleStudentMusicScores);
  263. }
  264. roomResult.setWhiteboards(whiteboardDao.findByRid(roomId));
  265. if (room != null) {
  266. roomResult.setSoundVolume(room.getSoundVolume());
  267. }
  268. log.info("join room: roomId = {}, userId = {}, userName={}, role = {}", roomId, userId, userName, roleEnum);
  269. return roomResult;
  270. }
  271. public RoomMember saveRoomMember(String roomId,String userId){
  272. SysUser sysUser = sysUserFeignService.queryUserById(Integer.parseInt(userId));
  273. Teacher teacher = teacherDao.get(Integer.parseInt(userId));
  274. CourseSchedule courseSchedule = courseScheduleDao.get(Long.parseLong(roomId.substring(1)));
  275. Date curTime = DateTimeUtils.currentUTC();
  276. RoleEnum roleEnum;
  277. RoomMember member = roomMemberDao.findByRidAndUid(roomId, userId);
  278. String userName;
  279. if (member == null) {
  280. if(teacher != null && teacher.getId().equals(courseSchedule.getActualTeacherId())){
  281. roleEnum = RoleTeacher;
  282. userName = sysUser.getRealName();
  283. }else {
  284. roleEnum = RoleStudent;
  285. userName = sysUser.getUsername();
  286. }
  287. member = saveRoomMember(userId, sysUser.getAvatar(), userName, roomId, roleEnum.getValue(), curTime);
  288. }
  289. return member;
  290. }
  291. @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
  292. @Override
  293. public void joinRoomFailure(String roomId, String userId) {
  294. RoomMember roomMember = roomMemberDao.findByRidAndUid(roomId, userId);
  295. if(roomMember == null){
  296. return ;
  297. }
  298. log.info("joinRoomFailure : roomId={}, userId={}", roomId, userId);
  299. //如果加入失败,删除该用户数据
  300. roomMemberDao.deleteUserByRidAndUid(roomId,userId);
  301. }
  302. @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
  303. @Override
  304. public void joinRoomSuccess(String roomId,String userId,String deviceNum) throws Exception {
  305. log.info("joinRoomSuccess: roomId={}, userId={}, deviceNum={}", roomId,userId,deviceNum);
  306. RoomMember roomMember = roomMemberDao.findByRidAndUid(roomId, userId);
  307. if(roomMember == null){
  308. roomMember = saveRoomMember(roomId,userId);
  309. }
  310. String joinSuccessKey = "joinRoomSuccess"+ roomId + userId;
  311. Boolean aBoolean = redisTemplate.opsForValue().setIfAbsent(joinSuccessKey, roomId, 1l, TimeUnit.SECONDS);
  312. if(!aBoolean){
  313. RoleEnum roleEnum = RoleEnum.getEnumByValue(roomMember.getRole());
  314. if(roleEnum == RoleTeacher && StringUtils.isNotEmpty(deviceNum)){
  315. teacherAttendanceService.updateDeviceNum(Integer.parseInt(roomId.substring(1)),userId,deviceNum,null);
  316. // signInSuccess(roomMember,deviceNum);
  317. }
  318. return;
  319. }
  320. RoleEnum roleEnum = RoleEnum.getEnumByValue(roomMember.getRole());
  321. CourseSchedule schedule = courseScheduleDao.getLock(Long.parseLong(roomId.substring(1)));
  322. String display = "";
  323. if(roleEnum == RoleTeacher){
  324. //如果是老师加入房间,调整节拍器状态
  325. courseScheduleStudentPaymentDao.adjustPlayMidi(schedule.getId(),null,null);
  326. display = "display://type=1?userId=" + userId + "?uri=";
  327. }else if (roleEnum == RoleEnum.RoleAssistant){
  328. display = "display://type=0?userId=" + userId + "?uri=";
  329. }
  330. Date curTime = DateTimeUtils.currentUTC();
  331. Room room = roomDao.findByRid(roomId);
  332. if (room == null) {
  333. saveRoom(roomId, roomId, curTime, display);
  334. IMApiResultInfo resultInfo = imHelper.createGroup(new String[]{userId}, roomId, roomId);
  335. if (!resultInfo.isSuccess()) {
  336. log.error("joinRoomSuccess createGroup error: roomId={}, {}", roomId, resultInfo.getErrorMessage());
  337. throw new ApiException(ErrorEnum.ERR_CREATE_ROOM_ERROR, resultInfo.getErrorMessage());
  338. }
  339. }else{
  340. if(roleEnum == RoleTeacher || roleEnum == RoleEnum.RoleAssistant){
  341. updateDisplay(roomId, userId, display, 0);
  342. }
  343. }
  344. UserInfo userInfo = userDao.findByUid(userId);
  345. if (userInfo == null) {
  346. userInfo = new UserInfo();
  347. userInfo.setUid(userId);
  348. userInfo.setName(roomMember.getName());
  349. userInfo.setCreateDt(curTime);
  350. userInfo.setUpdateDt(curTime);
  351. userDao.save(userInfo);
  352. }
  353. MemberChangedMessage msg = new MemberChangedMessage(MemberChangedMessage.Action_Join, userId, roleEnum.getValue());
  354. msg.setTimestamp(curTime);
  355. msg.setUserName(roomMember.getName());
  356. msg.setCamera(true);
  357. Boolean playMidi = false;
  358. Boolean examSong = false;
  359. if(roleEnum == RoleStudent){
  360. String midiByCourseIdAndUserId = courseScheduleStudentPaymentDao.getMidiByCourseIdAndUserId(schedule.getId().toString(), userId);
  361. //获取节拍器信息
  362. if(StringUtils.isNotEmpty(midiByCourseIdAndUserId)){
  363. JSONObject jsonObject = JSONObject.parseObject(midiByCourseIdAndUserId);
  364. if(jsonObject.get("enable") != null){
  365. playMidi = Boolean.parseBoolean(jsonObject.get("enable").toString());
  366. }
  367. }
  368. String examJson = courseScheduleStudentPaymentDao.getExamJsonByCourseIdAndUserId(schedule.getId(), Integer.parseInt(userId));
  369. if(StringUtils.isNotEmpty(examJson)){
  370. ExamSongDownloadData examSongDownloadData = JSON.parseObject(examJson, ExamSongDownloadData.class);
  371. examSong = examSongDownloadData.getEnable();
  372. }
  373. }
  374. msg.setHandUpOn(roomMember.isHand());
  375. msg.setMetronomeSwitch(playMidi);
  376. msg.setExamSongSwitch(examSong);
  377. imHelper.publishMessage(userId, roomId, msg);
  378. log.info("join room success: roomId = {}, userId = {}, role = {}", roomId, userId, roleEnum);
  379. signInSuccess(roomMember,deviceNum);
  380. }
  381. public void signInSuccess(RoomMember roomMember,String deviceNum) {
  382. String roomId = roomMember.getRid();
  383. String userId = roomMember.getUid();
  384. String currentRoomIdKey = roomId + userId;
  385. Long firstCourseId = Long.parseLong(roomId.substring(1));
  386. Long currentRoomId;
  387. if(redisTemplate.hasKey(currentRoomIdKey)){
  388. currentRoomId = Long.parseLong(redisTemplate.opsForValue().get(currentRoomIdKey));
  389. }else {
  390. log.error("signInFailure: roomId={}, userId={}", roomId, userId);
  391. currentRoomId = firstCourseId;
  392. }
  393. log.info("signInSuccess: roomId={}, userId={},currentRoomId={}", roomId, userId,currentRoomId);
  394. Integer userIdInt = Integer.parseInt(userId);
  395. RoleEnum roleEnum = RoleEnum.getEnumByValue(roomMember.getRole());
  396. if(roleEnum == RoleTeacher){
  397. teacherAttendanceService.addTeacherAttendanceSignIn(firstCourseId,userIdInt,currentRoomId,deviceNum);
  398. }else {
  399. studentAttendanceService.addStudentAttendanceSignIn(firstCourseId,userIdInt,currentRoomId);
  400. }
  401. redisTemplate.delete(roomId + userId);
  402. }
  403. public void saveRoom(String roomId, String roomName, Date createTime, String display) {
  404. Room room = new Room();
  405. room.setRid(roomId);
  406. room.setName(roomName);
  407. room.setCreateDt(createTime);
  408. room.setDisplay(display);
  409. room.setWhiteboardNameIndex(0);
  410. roomDao.save(room);
  411. }
  412. public RoomMember saveRoomMember(String userId, String headUrl, String userName, String roomId, int role, Date joinTime) {
  413. RoomMember roomMember = new RoomMember();
  414. roomMember.setUid(userId);
  415. roomMember.setName(userName);
  416. roomMember.setRid(roomId);
  417. roomMember.setRole(role);
  418. roomMember.setCamera(true);
  419. roomMember.setJoinDt(joinTime);
  420. roomMember.setMusicMode(false);
  421. roomMember.setHeadUrl(headUrl);
  422. roomMemberDao.save(roomMember);
  423. return roomMember;
  424. }
  425. @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class,isolation = Isolation.READ_COMMITTED)
  426. @Override
  427. public void leaveRoomSuccess(String roomId,String userId,String deviceNum) throws Exception {
  428. log.info("leaveRoomSuccess: roomId={}, userId={},deviceNum={}", roomId,userId,deviceNum);
  429. Integer firstCourseId = Integer.parseInt(roomId.substring(1));
  430. RoleEnum roleEnum;
  431. int parseInt = Integer.parseInt(userId);
  432. Teacher teacher = teacherDao.get(parseInt);
  433. CourseSchedule courseSchedule = courseScheduleDao.get(firstCourseId.longValue());
  434. if(teacher != null && teacher.getId().equals(courseSchedule.getActualTeacherId())){
  435. roleEnum = RoleTeacher;
  436. courseScheduleStudentMusicScoreDao.closePlayStatus(courseSchedule.getId(),null,null);
  437. }else {
  438. roleEnum = RoleStudent;
  439. }
  440. String leaveSuccessKey = "leaveRoomSuccess"+ roomId + userId;
  441. Boolean aBoolean = redisTemplate.opsForValue().setIfAbsent(leaveSuccessKey, roomId, 1l, TimeUnit.SECONDS);
  442. log.info("leaveRoomSuccess: roomId={}, userId={},deviceNum={},aBoolean={}", roomId,userId,deviceNum,aBoolean);
  443. if(!aBoolean){
  444. if (StringUtils.isNotEmpty(deviceNum)){
  445. //如果设备号不为空,更新设备号
  446. if(roleEnum == RoleTeacher){
  447. teacherAttendanceService.updateDeviceNum(firstCourseId,userId,null,deviceNum);
  448. // teacherAttendanceService.addTeacherAttendanceSignOut(firstCourseId.longValue(),parseInt,deviceNum);
  449. }
  450. }
  451. return;
  452. }
  453. String username;
  454. SysUser sysUser = sysUserFeignService.queryUserById(parseInt);
  455. if(roleEnum == RoleTeacher){
  456. username = sysUser.getRealName();
  457. courseScheduleStudentPaymentDao.adjustPlayMidi(firstCourseId,null,null);
  458. courseScheduleStudentPaymentDao.adjustExamSong(firstCourseId.longValue(),null,null);
  459. teacherAttendanceService.addTeacherAttendanceSignOut(firstCourseId.longValue(),parseInt,deviceNum);
  460. }else {
  461. username = sysUser.getUsername();
  462. studentAttendanceService.addStudentAttendanceSignOut(firstCourseId.longValue(),parseInt);
  463. }
  464. Room room = roomDao.findByRid(roomId);
  465. if (room == null) {
  466. roomMemberDao.deleteUserByRidAndUid(roomId, userId);
  467. userDao.deleteByUid(userId);
  468. return;
  469. }
  470. if (roleEnum != RoleStudent && isUserDisplay(room, userId)) {
  471. updateDisplay(roomId, userId, "", 0);
  472. }
  473. if (roomMemberDao.countByRid(roomId) <= 1) {
  474. roomMemberDao.deleteUserByRidAndUid(roomId, userId);
  475. roomDao.deleteByRid(roomId);
  476. deleteWhiteboardByUser(roomId, userId);
  477. imHelper.dismiss(userId, roomId);
  478. courseScheduleStudentMusicScoreDao.closePlayStatus(courseSchedule.getId(),null,null);
  479. log.info("leaveRoomSuccess dismiss the room: {},userId: {}", roomId,userId);
  480. } else {
  481. roomMemberDao.deleteUserByRidAndUid(roomId, userId);
  482. MemberChangedMessage msg = new MemberChangedMessage(MemberChangedMessage.Action_Leave, userId, roleEnum.getValue());
  483. msg.setUserName(username);
  484. imHelper.publishMessage(userId, roomId, msg);
  485. imHelper.quit(new String[]{userId}, roomId);
  486. log.info("leaveRoomSuccess quit group: roomId={},userId: {}", roomId,userId);
  487. }
  488. userDao.deleteByUid(userId);
  489. }
  490. public void deleteWhiteboardByUser(String roomId, String userId) throws Exception {
  491. List<Whiteboard> whiteboardList = whiteboardDao.findByRidAndCreator(roomId, userId);
  492. if (!whiteboardList.isEmpty()) {
  493. whiteboardDao.deleteByRidAndCreator(roomId, userId);
  494. for (Whiteboard wb : whiteboardList) {
  495. whiteBoardHelper.destroy(wb.getWbRoom());
  496. }
  497. }
  498. }
  499. @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
  500. @Override
  501. public void destroyRoom(String roomId) {
  502. whiteboardDao.deleteByRid(roomId);
  503. Room room = roomDao.findByRid(roomId);
  504. if (room == null) {
  505. List<RoomMember> list = roomMemberDao.findByRid(roomId);
  506. if (!list.isEmpty()) {
  507. try {
  508. imHelper.dismiss(list.get(0).getUid(), roomId);
  509. } catch (Exception e) {
  510. log.error("destroyRoom: {}", e.getMessage());
  511. e.printStackTrace();
  512. }
  513. }
  514. roomMemberDao.deleteByRid(roomId);
  515. log.info("destroyRoom: {}", roomId);
  516. }
  517. }
  518. @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
  519. @Override
  520. public Boolean downgrade(String roomId, List<ReqChangeUserRoleData.ChangedUser> users) throws Exception {
  521. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  522. CheckUtils.checkArgument(users.size() > 0, "the changed user list must't be null");
  523. SysUser authUser = sysUserFeignService.queryUserInfo();
  524. String userId = authUser.getId().toString();
  525. Room room = roomDao.findByRid(roomId);
  526. if (room == null) {
  527. throw new ApiException(ErrorEnum.ERR_ROOM_NOT_EXIST);
  528. }
  529. boolean result = false;
  530. List<RoleChangedMessage.ChangedUser> changedUsers = new ArrayList<>();
  531. for (ReqChangeUserRoleData.ChangedUser user : users) {
  532. String changedUserId = user.getUserId();
  533. RoleEnum changedRole = RoleEnum.getEnumByValue(user.getRole());
  534. if (changedUserId.equals(userId)) {
  535. log.error("can not change self role: {}, {}, {}", roomId, userId, changedRole);
  536. throw new ApiException(ErrorEnum.ERR_CHANGE_SELF_ROLE);
  537. } else {
  538. RoomMember oldUser = roomMemberDao.findByRidAndUid(roomId, changedUserId);
  539. if (oldUser != null) {
  540. if (changedRole.equals(RoleEnum.RoleAudience)) {
  541. int r = roomMemberDao.updateRoleByRidAndUid(roomId, changedUserId, changedRole.getValue());
  542. RoleChangedMessage.ChangedUser u = new RoleChangedMessage.ChangedUser(changedUserId, changedRole.getValue());
  543. UserInfo userInfo = userDao.findByUid(changedUserId);
  544. if (userInfo != null) {
  545. u.setUserName(userInfo.getName());
  546. }
  547. changedUsers.add(u);
  548. log.info("change the role: {}, {}, {}, result: {}", roomId, userId, changedRole, r);
  549. result = true;
  550. }
  551. if (oldUser.getRole() == RoleTeacher.getValue() && isUserDisplay(room, oldUser.getUid())) {
  552. updateDisplay(roomId, userId, "", 1);
  553. } else {
  554. log.info("don't update display: room={}, userRole={}", room, RoleEnum.getEnumByValue(oldUser.getRole()));
  555. }
  556. } else {
  557. log.info("role changed fail, not exist: {} - {} - {}", roomId, userId, changedRole);
  558. }
  559. }
  560. }
  561. if (result) {
  562. RoleChangedMessage msg = new RoleChangedMessage(userId);
  563. msg.setUsers(changedUsers);
  564. imHelper.publishMessage(userId, roomId, msg, 1);
  565. }
  566. return result;
  567. }
  568. @Override
  569. @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
  570. public Boolean kickMember(String roomId) throws Exception {
  571. SysUser authUser = sysUserFeignService.queryUserInfo();
  572. String userId = authUser.getId().toString();
  573. CheckUtils.checkArgument(userId != null, "userId must't be null");
  574. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  575. log.info("kickMember: roomId={}, userId={}", roomId, userId);
  576. RoomMember roomMember = roomMemberDao.findByRidAndUid(roomId, userId);
  577. if(roomMember == null){
  578. return true;
  579. }
  580. MemberChangedMessage msg = new MemberChangedMessage(MemberChangedMessage.Action_Kick, userId, roomMember.getRole());
  581. msg.setUserName(roomMember.getName());
  582. imHelper.publishMessage(userId, roomId, msg, 1);
  583. Room room = roomDao.findByRid(roomId);
  584. if (roomMember.getRole() == RoleTeacher.getValue() && isUserDisplay(room, userId)) {
  585. updateDisplay(roomId, userId, "", 1);
  586. }
  587. roomMemberDao.deleteUserByRidAndUid(roomId, userId);
  588. userDao.deleteByUid(userId);
  589. IMApiResultInfo apiResultInfo = imHelper.quit(new String[]{userId}, roomId);
  590. if (!apiResultInfo.isSuccess()) {
  591. throw new ApiException(ErrorEnum.ERR_EXIT_ROOM_ERROR, apiResultInfo.getErrorMessage());
  592. }
  593. return true;
  594. }
  595. @Override
  596. @Transactional(rollbackFor = Exception.class)
  597. public Boolean display(String roomId, int type, String uri, String targetUserId) throws Exception {
  598. SysUser authUser = sysUserFeignService.queryUserInfo();
  599. String userId = authUser.getId().toString();
  600. log.info("display in room: {}, type = {}, uri = {}", roomId, type, uri);
  601. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  602. CheckUtils.checkArgument(type >= 0 && type < DisplayEnum.values().length, "type not exist");
  603. DisplayEnum displayEnum = DisplayEnum.values()[type];
  604. if (displayEnum.equals(DisplayEnum.None)) {
  605. roomDao.updateDisplayByRid(roomId, "");
  606. DisplayMessage displayMessage = new DisplayMessage("");
  607. IMApiResultInfo apiResultInfo = imHelper.publishMessage(userId, roomId, displayMessage);
  608. if (apiResultInfo.isSuccess()) {
  609. return true;
  610. } else {
  611. throw new ApiException(ErrorEnum.ERR_MESSAGE_ERROR, apiResultInfo.getErrorMessage());
  612. }
  613. }
  614. String display = "display://type=" + type;
  615. if (displayEnum.equals(DisplayEnum.Teacher)) {
  616. List<RoomMember> teachers = roomMemberDao.findByRidAndRole(roomId, RoleTeacher.getValue());
  617. if (teachers.isEmpty()) {
  618. throw new ApiException(ErrorEnum.ERR_TEACHER_NOT_EXIST_IN_ROOM);
  619. } else {
  620. display += "?userId=" + teachers.get(0).getUid() + "?uri=";
  621. roomDao.updateDisplayByRid(roomId, display);
  622. DisplayMessage displayMessage = new DisplayMessage(display);
  623. imHelper.publishMessage(userId, roomId, displayMessage);
  624. log.info("change display to teacher: roomId={}, display={}", roomId, display);
  625. }
  626. } else if (displayEnum.equals(DisplayEnum.Assistant)) {
  627. List<RoomMember> assistants = roomMemberDao.findByRidAndRole(roomId, RoleEnum.RoleAssistant.getValue());
  628. if (assistants.isEmpty()) {
  629. throw new ApiException(ErrorEnum.ERR_ASSISTANT_NOT_EXIST_IN_ROOM);
  630. } else {
  631. display += "?userId=" + assistants.get(0).getUid() + "?uri=";
  632. roomDao.updateDisplayByRid(roomId, display);
  633. DisplayMessage displayMessage = new DisplayMessage(display);
  634. imHelper.publishMessage(userId, roomId, displayMessage);
  635. log.info("change display to assistant: roomId={}, display={}", roomId, display);
  636. }
  637. } else if (displayEnum.equals(DisplayEnum.Screen)) {
  638. display += "?userId=" + userId + "?uri=";
  639. roomDao.updateDisplayByRid(roomId, display);
  640. DisplayMessage displayMessage = new DisplayMessage(display);
  641. imHelper.publishMessage(userId, roomId, displayMessage);
  642. log.info("change display to screen: roomId={}, display={}", roomId, display);
  643. } else if (displayEnum.equals(DisplayEnum.STUDENT)) {
  644. display += "?userId=" + targetUserId + "?uri=" + uri;
  645. roomDao.updateDisplayByRid(roomId, display);
  646. DisplayMessage displayMessage = new DisplayMessage(display);
  647. imHelper.publishMessage(userId, roomId, displayMessage);
  648. log.info("change display to screen: roomId={}, display={}", roomId, display);
  649. } else {
  650. display += "?userId=" + userId + "?uri=" + uri;
  651. // CheckUtils.checkArgument(uri != null, "uri must't be null");
  652. // CheckUtils.checkArgument(whiteboardDao.findByRidAndWbid(roomId, uri).size() > 0, "whiteboard not exist");
  653. roomDao.updateDisplayByRid(roomId, display);
  654. DisplayMessage displayMessage = new DisplayMessage(display);
  655. imHelper.publishMessage(userId, roomId, displayMessage);
  656. }
  657. log.info("result display in room: {}, type = {}, uri = {}", roomId, type, uri);
  658. return true;
  659. }
  660. @Override
  661. @Transactional(rollbackFor = Exception.class)
  662. public String createWhiteBoard(String roomId) throws Exception {
  663. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  664. SysUser authUser = sysUserFeignService.queryUserInfo();
  665. String userId = authUser.getId().toString();
  666. log.info("createWhiteBoard: roomId = {}", roomId);
  667. String wbRoom = IdentifierUtils.uuid();
  668. WhiteBoardApiResultInfo resultInfo = whiteBoardHelper.create(wbRoom);
  669. if (resultInfo.isSuccess()) {
  670. String wbId = resultInfo.getData();
  671. Date date = DateTimeUtils.currentUTC();
  672. Room room = roomDao.findByRid(roomId);
  673. int whiteboardNameIndex = room.getWhiteboardNameIndex() + 1;
  674. String name = "白板" + whiteboardNameIndex;
  675. roomDao.updateWhiteboardNameIndexByRid(roomId, whiteboardNameIndex);
  676. Whiteboard wb = new Whiteboard();
  677. wb.setRid(roomId);
  678. wb.setWbRoom(wbRoom);
  679. wb.setWbid(wbId);
  680. wb.setName(name);
  681. wb.setCreator(userId);
  682. wb.setCreateDt(date);
  683. wb.setCurPg(0);
  684. whiteboardDao.save(wb);
  685. WhiteboardMessage wbmsg = new WhiteboardMessage(WhiteboardMessage.Create);
  686. wbmsg.setWhiteboardId(wbId);
  687. wbmsg.setWhiteboardName(name);
  688. imHelper.publishMessage(userId, roomId, wbmsg);
  689. String display = "display://type=2?userId=" + userId + "?uri=" + wbId;
  690. roomDao.updateDisplayByRid(roomId, display);
  691. DisplayMessage displayMessage = new DisplayMessage(display);
  692. imHelper.publishMessage(userId, roomId, displayMessage, 1);
  693. return wbId;
  694. } else {
  695. throw new ApiException(ErrorEnum.ERR_CREATE_WHITE_BOARD, resultInfo.getMsg());
  696. }
  697. }
  698. @Override
  699. @Transactional(rollbackFor = Exception.class)
  700. public Boolean deleteWhiteboard(String roomId, String whiteBoardId) throws Exception {
  701. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  702. CheckUtils.checkArgument(whiteBoardId != null, "whiteBoardId must't be null");
  703. SysUser authUser = sysUserFeignService.queryUserInfo();
  704. String userId = authUser.getId().toString();
  705. List<Whiteboard> whiteboardList = whiteboardDao.findByRidAndWbid(roomId, whiteBoardId);
  706. CheckUtils.checkArgument(whiteboardList.size() > 0, "whiteboard not exist");
  707. Room room = roomDao.findByRid(roomId);
  708. CheckUtils.checkArgument(room != null, "room not exist");
  709. log.info("deleteWhiteboard: room={}, whiteBoardId={}", room, whiteBoardId);
  710. String display = room.getDisplay();
  711. if (display.contains("uri=" + whiteBoardId)) {
  712. int result = roomDao.updateDisplayByRid(roomId, "");
  713. log.info("clear room display, room: {}, result: {}", roomId, result);
  714. DisplayMessage displayMessage = new DisplayMessage("");
  715. imHelper.publishMessage(userId, roomId, displayMessage, 1);
  716. } else {
  717. log.info("no display to clean: room={}", room);
  718. }
  719. String wbRoom = whiteboardList.get(0).getWbRoom();
  720. WhiteBoardApiResultInfo resultInfo = whiteBoardHelper.destroy(wbRoom);
  721. if (resultInfo.isSuccess()) {
  722. int result = whiteboardDao.deleteByWbid(whiteBoardId);
  723. log.info("delete whiteboard: roomId = {}, whiteBoardId = {}, result = {}", roomId, whiteBoardId, result);
  724. WhiteboardMessage wbmsg = new WhiteboardMessage(WhiteboardMessage.Delete);
  725. wbmsg.setWhiteboardId(whiteBoardId);
  726. imHelper.publishMessage(userId, roomId, wbmsg, 1);
  727. return true;
  728. } else {
  729. throw new ApiException(ErrorEnum.ERR_DELETE_WHITE_BOARD, resultInfo.getMsg());
  730. }
  731. }
  732. @Override
  733. @Transactional(rollbackFor = Exception.class)
  734. public List<RoomResult.WhiteboardResult> getWhiteboard(String roomId) throws Exception {
  735. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  736. CheckUtils.checkArgument(roomDao.existsByRid(roomId), "room not exist");
  737. List<Whiteboard> whiteboards = whiteboardDao.findByRid(roomId);
  738. List<RoomResult.WhiteboardResult> result = new ArrayList<>();
  739. for (Whiteboard wb : whiteboards) {
  740. RoomResult.WhiteboardResult r = new RoomResult.WhiteboardResult();
  741. r.setName(wb.getName());
  742. r.setCurPg(wb.getCurPg());
  743. r.setWhiteboardId(wb.getWbid());
  744. result.add(r);
  745. }
  746. return result;
  747. }
  748. @Override
  749. @Transactional(rollbackFor = Exception.class)
  750. public Boolean turnWhiteBoardPage(String roomId, String whiteBoardId, int page) throws Exception {
  751. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  752. CheckUtils.checkArgument(whiteBoardId != null, "whiteBoardId must't be null");
  753. SysUser authUser = sysUserFeignService.queryUserInfo();
  754. String userId = authUser.getId().toString();
  755. Room room = roomDao.findByRid(roomId);
  756. CheckUtils.checkArgument(room != null, "room not exist");
  757. int result = whiteboardDao.updatePageByRidAndWbid(roomId, whiteBoardId, page);
  758. log.info("turn page to: {}, room: {}, wb : {}; r: {}", page, roomId, whiteBoardId, result);
  759. TurnPageMessage turnPageMessage = new TurnPageMessage(whiteBoardId, userId, page);
  760. imHelper.publishMessage(userId, roomId, turnPageMessage);
  761. return true;
  762. }
  763. @Override
  764. @Transactional(rollbackFor = Exception.class)
  765. public Boolean controlDevice(ReqDeviceControlData data) throws Exception {
  766. String roomId = data.getRoomId();
  767. String userId = data.getUserId();
  768. DeviceTypeEnum typeEnum;
  769. boolean enable;
  770. if (data.getCameraOn() != null) {
  771. typeEnum = DeviceTypeEnum.Camera;
  772. enable = data.getCameraOn();
  773. } else if (data.getMicrophoneOn() != null) {
  774. typeEnum = DeviceTypeEnum.Microphone;
  775. enable = data.getMicrophoneOn();
  776. } else if (data.getMusicModeOn() != null) {
  777. typeEnum = DeviceTypeEnum.MusicMode;
  778. enable = data.getMusicModeOn();
  779. } else if (data.getHandUpOn() != null) {
  780. typeEnum = DeviceTypeEnum.HandUp;
  781. enable = data.getHandUpOn();
  782. }else if (data.getExamSongOn() != null) {
  783. typeEnum = DeviceTypeEnum.ExamSong;
  784. enable = data.getExamSongOn();
  785. }else if (data.getMusicScoreOn() != null) {
  786. typeEnum = DeviceTypeEnum.MusicScore;
  787. enable = data.getMusicScoreOn();
  788. if(enable){
  789. //保存伴奏音量
  790. roomDao.updateSoundVolumeById(roomId,data.getSoundVolume()==null?0:data.getSoundVolume());
  791. }
  792. }else if (data.getAccompanimentOn() != null) {
  793. typeEnum = DeviceTypeEnum.MusicScoreAccompaniment;
  794. enable = data.getAccompanimentOn();
  795. if(enable){
  796. //保存伴奏音量
  797. roomDao.updateSoundVolumeById(roomId,data.getSoundVolume()==null?0:data.getSoundVolume());
  798. }
  799. } else {
  800. throw new ApiException(ErrorEnum.ERR_REQUEST_PARA_ERR);
  801. }
  802. SysUser authUser = sysUserFeignService.queryUserInfo();
  803. log.info("controlDevice: userId = {}, typeEnum = {}, enable = {} ,roomId = {}", userId, typeEnum, enable, roomId);
  804. if (enable) {
  805. if (typeEnum.equals(DeviceTypeEnum.ExamSong)){
  806. long scheduleId = Long.parseLong(roomId.substring(1));
  807. ExamSongDownloadData msg;
  808. String examJson = courseScheduleStudentPaymentDao.getExamJsonByCourseIdAndUserId(scheduleId, Integer.parseInt(userId));
  809. if(StringUtils.isEmpty(examJson)){
  810. throw new BizException("学员伴奏信息异常");
  811. }else {
  812. msg = JSON.parseObject(examJson, ExamSongDownloadData.class);
  813. msg.setEnable(enable);
  814. }
  815. courseScheduleStudentPaymentDao.adjustExamSong(scheduleId,Integer.parseInt(userId),JSON.toJSONString(msg));
  816. DeviceStateChangedMessage deviceResourceMessage = new DeviceStateChangedMessage(typeEnum.ordinal(), enable);
  817. deviceResourceMessage.setUserId(userId);
  818. imHelper.publishMessage(authUser.getId().toString(), roomId, deviceResourceMessage, 1);
  819. }else if(typeEnum.equals(DeviceTypeEnum.MusicScore)){
  820. long scheduleId = Long.parseLong(roomId.substring(1));
  821. //关闭所有曲目播放
  822. courseScheduleStudentMusicScoreDao.closePlayStatus(scheduleId,Integer.parseInt(userId),null);
  823. DeviceStateChangedMessage deviceResourceMessage = new DeviceStateChangedMessage(typeEnum.ordinal(), enable);
  824. deviceResourceMessage.setMusicScoreAccompanimentId(data.getMusicScoreAccompanimentId());
  825. deviceResourceMessage.setUserId(userId);
  826. deviceResourceMessage.setSoundVolume(data.getSoundVolume());
  827. //原音
  828. courseScheduleStudentMusicScoreDao.openPlayStatus(scheduleId, data.getMusicScoreAccompanimentId(), Integer.parseInt(userId));
  829. imHelper.publishMessage(authUser.getId().toString(), roomId, deviceResourceMessage, 1);
  830. }else if(typeEnum.equals(DeviceTypeEnum.MusicScoreAccompaniment)){
  831. long scheduleId = Long.parseLong(roomId.substring(1));
  832. //关闭所有曲目播放
  833. courseScheduleStudentMusicScoreDao.closePlayStatus(scheduleId,Integer.parseInt(userId),null);
  834. courseScheduleStudentMusicScoreDao.openAccompanimentPlayStatus(scheduleId,data.getMusicScoreAccompanimentId(),Integer.parseInt(userId));
  835. DeviceStateChangedMessage deviceResourceMessage = new DeviceStateChangedMessage(typeEnum.ordinal(), enable);
  836. deviceResourceMessage.setMusicScoreAccompanimentId(data.getMusicScoreAccompanimentId());
  837. deviceResourceMessage.setUserId(userId);
  838. deviceResourceMessage.setSoundVolume(data.getSoundVolume());
  839. imHelper.publishMessage(authUser.getId().toString(), roomId, deviceResourceMessage, 1);
  840. }else {
  841. String ticket = IdentifierUtils.uuid();
  842. ControlDeviceTaskInfo taskInfo = new ControlDeviceTaskInfo();
  843. taskInfo.setRoomId(roomId);
  844. taskInfo.setTypeEnum(typeEnum);
  845. taskInfo.setOnOff(true);
  846. taskInfo.setApplyUserId(authUser.getId().toString());
  847. taskInfo.setTargetUserId(userId);
  848. taskInfo.setTicket(ticket);
  849. scheduleManager.addTask(taskInfo);
  850. ControlDeviceNotifyMessage msg = new ControlDeviceNotifyMessage(ActionEnum.Invite.ordinal());
  851. msg.setTicket(ticket);
  852. msg.setType(taskInfo.getTypeEnum().ordinal());
  853. msg.setOpUserId(authUser.getId().toString());
  854. msg.setOpUserName(authUser.getUsername());
  855. imHelper.publishMessage(authUser.getId().toString(), userId, roomId, msg);
  856. }
  857. } else {
  858. if (typeEnum.equals(DeviceTypeEnum.Camera)) {
  859. roomMemberDao.updateCameraByRidAndUid(roomId, userId, false);
  860. } else if (typeEnum.equals(DeviceTypeEnum.Microphone)){
  861. roomMemberDao.updateMicByRidAndUid(roomId, userId, false);
  862. } else if (typeEnum.equals(DeviceTypeEnum.HandUp)){
  863. roomMemberDao.updateHandByRidAndUid(roomId, userId, false);
  864. } else if (typeEnum.equals(DeviceTypeEnum.ExamSong)){
  865. long scheduleId = Long.parseLong(roomId.substring(1));
  866. ExamSongDownloadData msg;
  867. String examJson = courseScheduleStudentPaymentDao.getExamJsonByCourseIdAndUserId(scheduleId, Integer.parseInt(userId));
  868. if(StringUtils.isEmpty(examJson)){
  869. throw new BizException("学员伴奏信息异常");
  870. }else {
  871. msg = JSON.parseObject(examJson, ExamSongDownloadData.class);
  872. msg.setEnable(enable);
  873. }
  874. courseScheduleStudentPaymentDao.adjustExamSong(scheduleId,Integer.parseInt(userId),JSON.toJSONString(msg));
  875. }else if(typeEnum.equals(DeviceTypeEnum.MusicScore)){
  876. long scheduleId = Long.parseLong(roomId.substring(1));
  877. //关闭所有曲目播放
  878. courseScheduleStudentMusicScoreDao.closePlayStatus(scheduleId,Integer.parseInt(userId),null);
  879. }else if(typeEnum.equals(DeviceTypeEnum.MusicScoreAccompaniment)){
  880. long scheduleId = Long.parseLong(roomId.substring(1));
  881. //关闭所有曲目播放
  882. courseScheduleStudentMusicScoreDao.closePlayStatus(scheduleId,Integer.parseInt(userId),null);
  883. }else {
  884. roomMemberDao.updateMusicByRidAndUid(roomId, userId, false);
  885. }
  886. DeviceStateChangedMessage deviceResourceMessage = new DeviceStateChangedMessage(typeEnum.ordinal(), false);
  887. deviceResourceMessage.setUserId(userId);
  888. UserInfo userInfo = userDao.findByUid(userId);
  889. if (userInfo != null) {
  890. deviceResourceMessage.setUserName(userInfo.getName());
  891. }
  892. imHelper.publishMessage(authUser.getId().toString(), roomId, deviceResourceMessage, 1);
  893. }
  894. return true;
  895. }
  896. @Override
  897. @Transactional(rollbackFor = Exception.class)
  898. public Boolean batchControlDevice(ReqDeviceControlData data) throws Exception {
  899. if(data.getExamSongOn() != null || data.getMusicScoreOn() != null || data.getAccompanimentOn() != null){
  900. List<BasicUserDto> students = courseScheduleStudentPaymentDao.findStudents(Long.parseLong(data.getRoomId().substring(1)));
  901. for (BasicUserDto e:students) {
  902. data.setUserId(e.getUserId().toString());
  903. controlDevice(data);
  904. }
  905. return true;
  906. }
  907. List<RoomMember> roomMembers;
  908. if(StringUtils.isNotEmpty(data.getUserId())){
  909. roomMembers = new ArrayList<>();
  910. String[] split = data.getUserId().split(",");
  911. for (int i = 0; i < split.length; i++) {
  912. roomMembers.add(new RoomMember(split[i]));
  913. }
  914. }else {
  915. roomMembers = roomMemberDao.findByRidAndRole(data.getRoomId(), RoleStudent.getValue());
  916. }
  917. if(roomMembers.size() == 0){
  918. return false;
  919. }
  920. for (RoomMember e:roomMembers) {
  921. data.setUserId(e.getUid());
  922. controlDevice(data);
  923. }
  924. return true;
  925. }
  926. @Override
  927. @Transactional(rollbackFor = Exception.class)
  928. public Boolean approveControlDevice(ReqDeviceControlData data) throws Exception {
  929. String ticket = data.getTicket();
  930. String roomId = data.getRoomId();
  931. CheckUtils.checkArgument(ticket != null, "ticket must't be null");
  932. SysUser authUser = sysUserFeignService.queryUserInfo();
  933. String userId = authUser.getId().toString();
  934. log.info("approveControlDevice: ticket={}", ticket);
  935. ControlDeviceTaskInfo taskInfo = (ControlDeviceTaskInfo) scheduleManager.executeTask(ticket);
  936. if (taskInfo.getTypeEnum().equals(DeviceTypeEnum.Camera)) {
  937. roomMemberDao.updateCameraByRidAndUid(roomId, userId, taskInfo.isOnOff());
  938. }else if (taskInfo.getTypeEnum().equals(DeviceTypeEnum.ExamSong)) {
  939. long scheduleId = Long.parseLong(roomId.substring(1));
  940. ExamSongDownloadData msg;
  941. String examJson = courseScheduleStudentPaymentDao.getExamJsonByCourseIdAndUserId(scheduleId, authUser.getId());
  942. if(StringUtils.isEmpty(examJson)){
  943. throw new BizException("学员伴奏信息异常");
  944. }else {
  945. msg = JSON.parseObject(examJson, ExamSongDownloadData.class);
  946. msg.setEnable(true);
  947. }
  948. courseScheduleStudentPaymentDao.adjustExamSong(scheduleId,authUser.getId(),JSON.toJSONString(msg));
  949. }else {
  950. roomMemberDao.updateMicByRidAndUid(roomId, userId, taskInfo.isOnOff());
  951. }
  952. ControlDeviceNotifyMessage msg = new ControlDeviceNotifyMessage(ActionEnum.Approve.ordinal());
  953. msg.setType(taskInfo.getTypeEnum().ordinal());
  954. msg.setOpUserId(userId);
  955. msg.setOpUserName(authUser.getUsername());
  956. imHelper.publishMessage(userId, taskInfo.getApplyUserId(), roomId, msg);
  957. DeviceStateChangedMessage deviceResourceMessage = new DeviceStateChangedMessage(taskInfo.getTypeEnum().ordinal(), taskInfo.isOnOff());
  958. deviceResourceMessage.setUserId(userId);
  959. imHelper.publishMessage(userId, roomId, deviceResourceMessage, 1);
  960. return true;
  961. }
  962. @Override
  963. @Transactional(rollbackFor = Exception.class)
  964. public Boolean rejectControlDevice(String roomId, String ticket) throws Exception {
  965. CheckUtils.checkArgument(ticket != null, "ticket must't be null");
  966. SysUser authUser = sysUserFeignService.queryUserInfo();
  967. String userId = authUser.getId().toString();
  968. log.info("rejectControlDevice: ticket={}", ticket);
  969. ControlDeviceTaskInfo taskInfo = (ControlDeviceTaskInfo) scheduleManager.executeTask(ticket);
  970. ControlDeviceNotifyMessage msg = new ControlDeviceNotifyMessage(ActionEnum.Reject.ordinal());
  971. msg.setType(taskInfo.getTypeEnum().ordinal());
  972. msg.setOpUserId(userId);
  973. msg.setOpUserName(authUser.getUsername());
  974. imHelper.publishMessage(userId, taskInfo.getApplyUserId(), roomId, msg);
  975. return true;
  976. }
  977. @Override
  978. @Transactional(rollbackFor = Exception.class)
  979. public Boolean syncDeviceState(ReqDeviceControlData data) throws Exception {
  980. String roomId = data.getRoomId();
  981. DeviceTypeEnum type;
  982. boolean enable;
  983. if (data.getCameraOn() != null) {
  984. type = DeviceTypeEnum.Camera;
  985. enable = data.getCameraOn();
  986. } else if (data.getMicrophoneOn() != null) {
  987. type = DeviceTypeEnum.Microphone;
  988. enable = data.getMicrophoneOn();
  989. } else if (data.getMusicModeOn() != null) {
  990. type = DeviceTypeEnum.MusicMode;
  991. enable = data.getMusicModeOn();
  992. } else if (data.getHandUpOn() != null) {
  993. type = DeviceTypeEnum.HandUp;
  994. enable = data.getHandUpOn();
  995. } else if (data.getExamSongOn() != null) {
  996. type = DeviceTypeEnum.ExamSong;
  997. enable = data.getExamSongOn();
  998. }else {
  999. throw new ApiException(ErrorEnum.ERR_REQUEST_PARA_ERR);
  1000. }
  1001. SysUser authUser = sysUserFeignService.queryUserInfo();
  1002. String userId = authUser.getId().toString();
  1003. DeviceStateChangedMessage deviceResourceMessage = new DeviceStateChangedMessage(type.ordinal(), enable);
  1004. if (type.equals(DeviceTypeEnum.Camera)) {
  1005. roomMemberDao.updateCameraByRidAndUid(roomId, userId, enable);
  1006. } else if (type.equals(DeviceTypeEnum.Microphone)){
  1007. roomMemberDao.updateMicByRidAndUid(roomId, userId, enable);
  1008. } else if (type.equals(DeviceTypeEnum.HandUp)){
  1009. roomMemberDao.updateHandByRidAndUid(roomId, userId, enable);
  1010. } else if (type.equals(DeviceTypeEnum.ExamSong)){
  1011. long scheduleId = Long.parseLong(roomId.substring(1));
  1012. ExamSongDownloadData msg;
  1013. String examJson = courseScheduleStudentPaymentDao.getExamJsonByCourseIdAndUserId(scheduleId, authUser.getId());
  1014. if(StringUtils.isEmpty(examJson)){
  1015. throw new BizException("学员伴奏信息异常");
  1016. }else {
  1017. msg = JSON.parseObject(examJson, ExamSongDownloadData.class);
  1018. msg.setEnable(enable);
  1019. }
  1020. courseScheduleStudentPaymentDao.adjustExamSong(scheduleId,authUser.getId(),JSON.toJSONString(msg));
  1021. }else {
  1022. roomMemberDao.updateMusicByRidAndUid(roomId, userId, enable);
  1023. }
  1024. Room room = roomDao.findByRid(roomId);
  1025. deviceResourceMessage.setSoundVolume(room.getSoundVolume());
  1026. deviceResourceMessage.setUserId(userId);
  1027. imHelper.publishMessage(userId, roomId, deviceResourceMessage, 1);
  1028. log.info("syncDeviceState : {}, {}", roomId, enable);
  1029. return true;
  1030. }
  1031. @Override
  1032. @Transactional(rollbackFor = Exception.class)
  1033. public List<RoomResult.MemberResult> getMembers(String roomId) throws Exception {
  1034. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  1035. List<RoomMember> roomMemberList = roomMemberDao.findByRid(roomId);
  1036. if(roomMemberList != null && roomMemberList.size() > 0){
  1037. List<CourseScheduleStudentMusicScore> scheduleStudentMusicScores = courseScheduleStudentMusicScoreDao.queryByScoreIdAndCourseId(null, Long.parseLong(roomId.substring(1)), null, null, null);
  1038. RoomResult roomResult = new RoomResult();
  1039. Set<String> userIds = roomMemberList.stream().map(e -> e.getUid()).collect(Collectors.toSet());
  1040. Map<Integer,String> midiMap = MapUtil.convertMybatisMap(courseScheduleStudentPaymentDao.queryMidiByUserIdsAndCourseId(userIds,roomId.substring(1)));
  1041. Map<Integer,String> examSongMap = MapUtil.convertMybatisMap(courseScheduleStudentPaymentDao.queryExamSongByUserIdsAndCourseId(userIds,roomId.substring(1)));
  1042. roomResult.setMembers(roomMemberList,midiMap,examSongMap,scheduleStudentMusicScores);
  1043. return roomResult.getMembers();
  1044. }
  1045. return null;
  1046. }
  1047. @Override
  1048. @Transactional(rollbackFor = Exception.class)
  1049. public Boolean applySpeech(String roomId) throws Exception {
  1050. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  1051. CheckUtils.checkArgument(roomDao.existsByRid(roomId), "room not exist");
  1052. SysUser authUser = sysUserFeignService.queryUserInfo();
  1053. String userId = authUser.getId().toString();
  1054. List<RoomMember> assistants = roomMemberDao.findByRidAndRole(roomId, RoleEnum.RoleAssistant.getValue());
  1055. if (assistants.isEmpty()) {
  1056. throw new ApiException(ErrorEnum.ERR_ASSISTANT_NOT_EXIST_IN_ROOM);
  1057. }
  1058. String ticket = IdentifierUtils.uuid();
  1059. ScheduledTaskInfo scheduledTaskInfo = new ScheduledTaskInfo();
  1060. scheduledTaskInfo.setTicket(ticket);
  1061. scheduledTaskInfo.setRoomId(roomId);
  1062. scheduledTaskInfo.setApplyUserId(userId);
  1063. scheduledTaskInfo.setTargetUserId(assistants.get(0).getUid());
  1064. scheduleManager.addTask(scheduledTaskInfo);
  1065. log.info("applySpeech: task = {}", scheduledTaskInfo);
  1066. ApplyForSpeechMessage msg = new ApplyForSpeechMessage();
  1067. msg.setTicket(ticket);
  1068. msg.setReqUserId(userId);
  1069. IMApiResultInfo resultInfo = imHelper.publishMessage(userId, assistants.get(0).getUid(), roomId, msg);
  1070. log.info("apply for speech: {}, task = {}", roomId, scheduledTaskInfo);
  1071. if (resultInfo.isSuccess()) {
  1072. return true;
  1073. } else {
  1074. throw new ApiException(ErrorEnum.ERR_MESSAGE_ERROR, resultInfo.getErrorMessage());
  1075. }
  1076. }
  1077. @Override
  1078. @Transactional(rollbackFor = Exception.class)
  1079. public Boolean approveSpeech(String roomId, String ticket) throws Exception {
  1080. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  1081. CheckUtils.checkArgument(roomDao.existsByRid(roomId), "room not exist");
  1082. SysUser authUser = sysUserFeignService.queryUserInfo();
  1083. String userId = authUser.getId().toString();
  1084. int count = roomMemberDao.countByRidAndExcludeRole(roomId, RoleEnum.RoleAudience.getValue());
  1085. if (count == roomProperties.getMaxCount()) {
  1086. log.error("approveSpeech error: roomId = {}, ticket={}", roomId, ticket);
  1087. throw new ApiException(ErrorEnum.ERR_OVER_MAX_COUNT);
  1088. }
  1089. ScheduledTaskInfo taskInfo = scheduleManager.executeTask(ticket);
  1090. log.info("approveSpeech: task = {}", taskInfo);
  1091. roomMemberDao.updateRoleByRidAndUid(roomId, taskInfo.getApplyUserId(), RoleStudent.getValue());
  1092. SpeechResultMessage msg = new SpeechResultMessage(SpeechResultMessage.Action_Approve);
  1093. UserInfo userInfo = userDao.findByUid(taskInfo.getApplyUserId());
  1094. msg.setOpUserId(userId);
  1095. msg.setOpUserName(authUser.getUsername());
  1096. msg.setReqUserId(taskInfo.getApplyUserId());
  1097. if (userInfo != null) {
  1098. msg.setReqUserName(userInfo.getName());
  1099. }
  1100. msg.setRole(RoleStudent.getValue());
  1101. IMApiResultInfo resultInfo = imHelper.publishMessage(userId, taskInfo.getApplyUserId(), roomId, msg);
  1102. if (!resultInfo.isSuccess()) {
  1103. throw new ApiException(ErrorEnum.ERR_MESSAGE_ERROR, resultInfo.getErrorMessage());
  1104. }
  1105. RoleChangedMessage rcMsg = new RoleChangedMessage(userId);
  1106. List<RoleChangedMessage.ChangedUser> changedUserList = new ArrayList<>();
  1107. RoleChangedMessage.ChangedUser user = new RoleChangedMessage.ChangedUser(taskInfo.getApplyUserId(), RoleStudent.getValue());
  1108. if (userInfo != null) {
  1109. user.setUserName(userInfo.getName());
  1110. }
  1111. changedUserList.add(user);
  1112. rcMsg.setUsers(changedUserList);
  1113. imHelper.publishMessage(userId, roomId, rcMsg, 1);
  1114. return true;
  1115. }
  1116. @Override
  1117. @Transactional(rollbackFor = Exception.class)
  1118. public Boolean rejectSpeech(String roomId, String ticket) throws Exception {
  1119. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  1120. CheckUtils.checkArgument(roomDao.existsByRid(roomId), "room not exist");
  1121. SysUser authUser = sysUserFeignService.queryUserInfo();
  1122. String userId = authUser.getId().toString();
  1123. ScheduledTaskInfo taskInfo = scheduleManager.executeTask(ticket);
  1124. log.info("rejectSpeech: task = {}", taskInfo);
  1125. SpeechResultMessage msg = new SpeechResultMessage(SpeechResultMessage.Action_Reject);
  1126. msg.setOpUserId(userId);
  1127. msg.setOpUserName(authUser.getUsername());
  1128. msg.setRole(RoleStudent.getValue());
  1129. IMApiResultInfo resultInfo = imHelper.publishMessage(userId, taskInfo.getApplyUserId(), roomId, msg);
  1130. if (resultInfo.isSuccess()) {
  1131. return true;
  1132. } else {
  1133. throw new ApiException(ErrorEnum.ERR_MESSAGE_ERROR, resultInfo.getErrorMessage());
  1134. }
  1135. }
  1136. public void checkOverMax(String roomId, RoomMember targetUser, int targetRole) {
  1137. if (RoleEnum.getEnumByValue(targetUser.getRole()).equals(RoleEnum.RoleAudience)) {
  1138. int count = roomMemberDao.countByRidAndExcludeRole(roomId, RoleEnum.RoleAudience.getValue());
  1139. if (count == roomProperties.getMaxCount()) {
  1140. log.error("assign error: roomId = {}, userId = {}, role = {}", roomId, targetUser.getRid(), targetUser.getRole());
  1141. throw new ApiException(ErrorEnum.ERR_OVER_MAX_COUNT);
  1142. }
  1143. } else if (targetRole > targetUser.getRole()) {
  1144. throw new ApiException(ErrorEnum.ERR_DOWNGRADE_ROLE);
  1145. }
  1146. }
  1147. @Override
  1148. @Transactional(rollbackFor = Exception.class)
  1149. public Boolean transfer(String roomId, String userId) throws Exception {
  1150. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  1151. CheckUtils.checkArgument(userId != null, "userId must't be null");
  1152. CheckUtils.checkArgument(!userId.equals(userId), "can't set self role");
  1153. log.info("transfer: roomId = {}, userId = {}", roomId, userId);
  1154. RoomMember roomMember = roomMemberDao.findByRidAndUid(roomId, userId);
  1155. if (roomMember == null) {
  1156. log.error("assistant transfer error: {} toUser = {}, opUser={}", roomId, userId, userId);
  1157. throw new ApiException(ErrorEnum.ERR_USER_NOT_EXIST_IN_ROOM);
  1158. }
  1159. Room room = roomDao.findByRid(roomId);
  1160. if (room == null) {
  1161. log.error("assistant transfer error: {} toUser = {}, opUser={}", roomId, userId, userId);
  1162. throw new ApiException(ErrorEnum.ERR_ROOM_NOT_EXIST);
  1163. }
  1164. if (isUserDisplay(room, userId) || isUserDisplay(room, userId)) {
  1165. updateDisplay(roomId, userId, "", 1);
  1166. } else {
  1167. log.info("don't update display: room={}", room);
  1168. }
  1169. roomMemberDao.updateRoleByRidAndUid(roomId, userId, RoleStudent.getValue());
  1170. roomMemberDao.updateRoleByRidAndUid(roomId, userId, RoleEnum.RoleAssistant.getValue());
  1171. AssistantTransferMessage msg = new AssistantTransferMessage();
  1172. msg.setOpUserId(userId);
  1173. msg.setToUserId(userId);
  1174. IMApiResultInfo resultInfo = imHelper.publishMessage(userId, roomId, msg, 1);
  1175. if (resultInfo.isSuccess()) {
  1176. return true;
  1177. } else {
  1178. throw new ApiException(ErrorEnum.ERR_MESSAGE_ERROR, resultInfo.getErrorMessage());
  1179. }
  1180. }
  1181. @Override
  1182. @Transactional(rollbackFor = Exception.class)
  1183. public Boolean inviteUpgradeRole(String roomId, String targetUserId, int targetRole) throws Exception {
  1184. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  1185. CheckUtils.checkArgument(targetUserId != null, "userId must't be null");
  1186. CheckUtils.checkArgument(roomMemberDao.existsByRidAndUid(roomId, targetUserId), "room member not exist");
  1187. SysUser authUser = sysUserFeignService.queryUserInfo();
  1188. String userId = authUser.getId().toString();
  1189. log.info("inviteUpgradeRole roomId = {}, targetUserId = {}, targetRole = {}", roomId, targetUserId, targetRole);
  1190. RoomMember targetUser = roomMemberDao.findByRidAndUid(roomId, targetUserId);
  1191. if (targetUser == null) {
  1192. throw new ApiException(ErrorEnum.ERR_USER_NOT_EXIST_IN_ROOM);
  1193. }
  1194. checkOverMax(roomId, targetUser, targetRole);
  1195. String ticket = IdentifierUtils.uuid();
  1196. UpgradeRoleTaskInfo taskInfo = new UpgradeRoleTaskInfo();
  1197. taskInfo.setTicket(ticket);
  1198. taskInfo.setRoomId(roomId);
  1199. taskInfo.setApplyUserId(userId);
  1200. taskInfo.setTargetUserId(targetUserId);
  1201. taskInfo.setRole(RoleEnum.getEnumByValue(targetRole));
  1202. scheduleManager.addTask(taskInfo);
  1203. UpgradeRoleMessage msg = new UpgradeRoleMessage(ActionEnum.Invite.ordinal());
  1204. msg.setTicket(ticket);
  1205. msg.setOpUserId(userId);
  1206. msg.setOpUserName(authUser.getUsername());
  1207. msg.setRole(targetRole);
  1208. IMApiResultInfo resultInfo = imHelper.publishMessage(userId, targetUserId, roomId, msg);
  1209. if (resultInfo.isSuccess()) {
  1210. return true;
  1211. } else {
  1212. throw new ApiException(ErrorEnum.ERR_MESSAGE_ERROR, resultInfo.getErrorMessage());
  1213. }
  1214. }
  1215. @Override
  1216. @Transactional(rollbackFor = Exception.class)
  1217. public Boolean approveUpgradeRole(String roomId, String ticket) throws Exception {
  1218. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  1219. CheckUtils.checkArgument(ticket != null, "ticket must't be null");
  1220. CheckUtils.checkArgument(roomDao.existsByRid(roomId), "room not exist");
  1221. SysUser authUser = sysUserFeignService.queryUserInfo();
  1222. String userId = authUser.getId().toString();
  1223. UpgradeRoleTaskInfo taskInfo = (UpgradeRoleTaskInfo) scheduleManager.executeTask(ticket);
  1224. log.info("approveUpgradeRole roomId = {}, task={}", roomId, taskInfo);
  1225. RoomMember targetUser = roomMemberDao.findByRidAndUid(roomId, userId);
  1226. if (targetUser == null) {
  1227. throw new ApiException(ErrorEnum.ERR_USER_NOT_EXIST_IN_ROOM);
  1228. }
  1229. if (!taskInfo.getTargetUserId().equals(userId)) {
  1230. throw new ApiException(ErrorEnum.ERR_APPLY_TICKET_INVALID);
  1231. }
  1232. checkOverMax(roomId, targetUser, taskInfo.getRole().getValue());
  1233. roomMemberDao.updateRoleByRidAndUid(roomId, userId, taskInfo.getRole().getValue());
  1234. UpgradeRoleMessage msg = new UpgradeRoleMessage(ActionEnum.Approve.ordinal());
  1235. msg.setOpUserName(authUser.getUsername());
  1236. msg.setOpUserId(userId);
  1237. msg.setRole(taskInfo.getRole().getValue());
  1238. IMApiResultInfo resultInfo = imHelper.publishMessage(userId, taskInfo.getApplyUserId(), roomId, msg);
  1239. if (!resultInfo.isSuccess()) {
  1240. throw new ApiException(ErrorEnum.ERR_MESSAGE_ERROR, resultInfo.getErrorMessage());
  1241. }
  1242. RoleChangedMessage rcMsg = new RoleChangedMessage(userId);
  1243. List<RoleChangedMessage.ChangedUser> changedUserList = new ArrayList<>();
  1244. RoleChangedMessage.ChangedUser user = new RoleChangedMessage.ChangedUser(userId, taskInfo.getRole().getValue());
  1245. user.setUserName(authUser.getUsername());
  1246. changedUserList.add(user);
  1247. rcMsg.setUsers(changedUserList);
  1248. imHelper.publishMessage(userId, roomId, rcMsg, 1);
  1249. return true;
  1250. }
  1251. @Override
  1252. @Transactional(rollbackFor = Exception.class)
  1253. public Boolean rejectUpgradeRole(String roomId, String ticket) throws Exception {
  1254. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  1255. CheckUtils.checkArgument(ticket != null, "ticket must't be null");
  1256. CheckUtils.checkArgument(roomDao.existsByRid(roomId), "room not exist");
  1257. SysUser authUser = sysUserFeignService.queryUserInfo();
  1258. String userId = authUser.getId().toString();
  1259. UpgradeRoleTaskInfo taskInfo = (UpgradeRoleTaskInfo) scheduleManager.executeTask(ticket);
  1260. UpgradeRoleMessage msg = new UpgradeRoleMessage(ActionEnum.Reject.ordinal());
  1261. msg.setOpUserName(authUser.getUsername());
  1262. msg.setOpUserId(userId);
  1263. msg.setRole(taskInfo.getRole().getValue());
  1264. IMApiResultInfo resultInfo = imHelper.publishMessage(userId, taskInfo.getApplyUserId(), roomId, msg);
  1265. if (resultInfo.isSuccess()) {
  1266. return true;
  1267. } else {
  1268. throw new ApiException(ErrorEnum.ERR_MESSAGE_ERROR, resultInfo.getErrorMessage());
  1269. }
  1270. }
  1271. @Override
  1272. @Transactional(rollbackFor = Exception.class)
  1273. public Boolean changeRole(String roomId, String targetUserId, int targetRole) throws Exception {
  1274. CheckUtils.checkArgument(roomId != null, "roomId must't be null");
  1275. CheckUtils.checkArgument(targetUserId != null, "userId must't be null");
  1276. CheckUtils.checkArgument(roomDao.existsByRid(roomId), "room not exist");
  1277. CheckUtils.checkArgument(RoleEnum.getEnumByValue(targetRole).equals(RoleTeacher), "only set to teacher");
  1278. CheckUtils.checkArgument(roomMemberDao.existsByRidAndUid(roomId, targetUserId), "room member not exist");
  1279. SysUser authUser = sysUserFeignService.queryUserInfo();
  1280. String userId = authUser.getId().toString();
  1281. RoomMember targetUser = roomMemberDao.findByRidAndUid(roomId, targetUserId);
  1282. if (targetUser == null) {
  1283. throw new ApiException(ErrorEnum.ERR_USER_NOT_EXIST_IN_ROOM);
  1284. } else {
  1285. if (!RoleEnum.getEnumByValue(targetUser.getRole()).equals(RoleStudent)) {
  1286. log.error("change role error: targetUserId={}, targetRole = {}", targetUser, RoleEnum.getEnumByValue(targetRole));
  1287. throw new ApiException(ErrorEnum.ERR_CHANGE_ROLE);
  1288. }
  1289. }
  1290. log.info("changeRole: roomId={}, targetUserId={}", roomId, targetUserId);
  1291. List<RoleChangedMessage.ChangedUser> changedUserList = new ArrayList<>();
  1292. RoleChangedMessage msg = new RoleChangedMessage(userId);
  1293. List<RoomMember> teachers = roomMemberDao.findByRidAndRole(roomId, RoleTeacher.getValue());
  1294. if (!teachers.isEmpty()) {
  1295. roomMemberDao.updateRoleByRidAndUid(roomId, teachers.get(0).getUid(), RoleStudent.getValue());
  1296. RoleChangedMessage.ChangedUser user = new RoleChangedMessage.ChangedUser(teachers.get(0).getUid(), RoleStudent.getValue());
  1297. UserInfo userInfo = userDao.findByUid(teachers.get(0).getUid());
  1298. if (userInfo != null) {
  1299. user.setUserName(userInfo.getName());
  1300. }
  1301. changedUserList.add(user);
  1302. } else {
  1303. log.info("change directly cause no teacher exist in room, roomId={}", roomId);
  1304. }
  1305. roomMemberDao.updateRoleByRidAndUid(roomId, targetUserId, targetRole);
  1306. RoleChangedMessage.ChangedUser user = new RoleChangedMessage.ChangedUser(targetUserId, targetRole);
  1307. UserInfo userInfo = userDao.findByUid(targetUserId);
  1308. if (userInfo != null) {
  1309. user.setUserName(userInfo.getName());
  1310. }
  1311. changedUserList.add(user);
  1312. msg.setUsers(changedUserList);
  1313. imHelper.publishMessage(userId, roomId, msg, 1);
  1314. String display = "display://type=1?userId=" + targetUserId + "?uri=";
  1315. DisplayMessage displayMessage = new DisplayMessage(display);
  1316. roomDao.updateDisplayByRid(roomId, display);
  1317. imHelper.publishMessage(userId, roomId, displayMessage, 1);
  1318. log.info("changeRole, display changed: roomId={}, {}, targetUserId={}", roomId, display, targetUserId);
  1319. return true;
  1320. }
  1321. @Override
  1322. @Transactional(rollbackFor = Exception.class)
  1323. public Boolean memberOnlineStatus(List<ReqMemberOnlineStatus> statusList, String nonce, String timestamp, String signature) throws Exception {
  1324. String sign = imProperties.getSecret() + nonce + timestamp;
  1325. String signSHA1 = CodeUtil.hexSHA1(sign);
  1326. if (!signSHA1.equals(signature)) {
  1327. log.info("memberOnlineStatus signature error");
  1328. return true;
  1329. }
  1330. for (ReqMemberOnlineStatus status : statusList) {
  1331. int s = Integer.parseInt(status.getStatus());
  1332. String userId = status.getUserId();
  1333. log.info("memberOnlineStatus, userId={}, status={}", userId, status);
  1334. //1:offline 离线; 0: online 在线
  1335. if (s == 1) {
  1336. List<RoomMember> members = roomMemberDao.findByUid(userId);
  1337. if (!members.isEmpty()) {
  1338. scheduleManager.userIMOffline(userId);
  1339. }
  1340. } else if (s == 0) {
  1341. scheduleManager.userIMOnline(userId);
  1342. }
  1343. }
  1344. return true;
  1345. }
  1346. @Override
  1347. @Transactional(rollbackFor = Exception.class)
  1348. public void userIMOfflineKick(String userId) {
  1349. List<RoomMember> members = roomMemberDao.findByUid(userId);
  1350. for (RoomMember member : members) {
  1351. int userRole = member.getRole();
  1352. log.info("userIMOfflineKick: roomId={}, {}, role={}", member.getRid(), userId, RoleEnum.getEnumByValue(userRole));
  1353. try {
  1354. if (userRole == RoleTeacher.getValue() || userRole == RoleEnum.RoleAssistant.getValue()) {
  1355. Room room = roomDao.findByRid(member.getRid());
  1356. if (room == null) {
  1357. break;
  1358. }
  1359. if (isUserDisplay(room, member.getUid())) {
  1360. updateDisplay(member.getRid(), member.getUid(), "", 0);
  1361. log.info("memberOnlineStatus offline: roomId={}, {}", member.getRid(), member.getUid());
  1362. }
  1363. }
  1364. if (roomMemberDao.countByRid(member.getRid()) == 1) {
  1365. IMApiResultInfo apiResultInfo = null;
  1366. apiResultInfo = imHelper.dismiss(member.getUid(), member.getRid());
  1367. if (apiResultInfo.getCode() == 200) {
  1368. roomMemberDao.deleteUserByRidAndUid(member.getRid(), member.getUid());
  1369. roomDao.deleteByRid(member.getRid());
  1370. deleteWhiteboardByUser(member.getRid(), member.getUid());
  1371. log.info("dismiss the room: {},userId: {}", member.getRid(),userId);
  1372. } else {
  1373. log.error("{} exit {} room error: {}", member.getUid(), member.getRid(), apiResultInfo.getErrorMessage());
  1374. }
  1375. } else {
  1376. IMApiResultInfo apiResultInfo = null;
  1377. apiResultInfo = imHelper.quit(new String[]{member.getUid()}, member.getRid());
  1378. if (apiResultInfo.isSuccess()) {
  1379. roomMemberDao.deleteUserByRidAndUid(member.getRid(), member.getUid());
  1380. MemberChangedMessage msg = new MemberChangedMessage(MemberChangedMessage.Action_Leave, member.getUid(), userRole);
  1381. msg.setUserName(member.getName());
  1382. imHelper.publishMessage(member.getUid(), member.getRid(), msg);
  1383. imHelper.quit(new String[]{member.getUid()}, member.getRid());
  1384. log.info("quit group: roomId={},userId: {}", member.getRid(), member.getUid());
  1385. } else {
  1386. log.error("{} exit {} room error: {}", member.getUid(), member.getRid(), apiResultInfo.getErrorMessage());
  1387. }
  1388. }
  1389. userDao.deleteByUid(member.getUid());
  1390. } catch (Exception e) {
  1391. log.error("userIMOfflineKick error: userId={}", userId);
  1392. }
  1393. }
  1394. }
  1395. @Override
  1396. @Transactional(rollbackFor = Exception.class)
  1397. public void sendImPlayMidiMessage(PlayMidiMessageData playMidiMessageData) throws Exception {
  1398. SysUser sysUser = sysUserFeignService.queryUserInfo();
  1399. if(sysUser == null){
  1400. throw new BizException("用户信息获取失败");
  1401. }
  1402. String content = playMidiMessageData.getContent();
  1403. String roomId = playMidiMessageData.getRoomId();
  1404. CustomMessage customMessage = JSONObject.parseObject(content,CustomMessage.class);
  1405. String userId = sysUser.getId().toString();
  1406. MetronomeMessageMessage displayMessage = new MetronomeMessageMessage(customMessage);
  1407. imHelper.publishMessage(userId, roomId, displayMessage, 1);
  1408. //记录节拍器消息
  1409. courseScheduleStudentPaymentDao.adjustPlayMidi(Long.parseLong(roomId.substring(1)),playMidiMessageData.getUserId(),content);
  1410. }
  1411. @Override
  1412. @Transactional(rollbackFor = Exception.class)
  1413. public void pushDownloadMusicScoreMsg(MusicScoreData musicScoreData) throws Exception {
  1414. SysUser authUser = sysUserFeignService.queryUserInfo();
  1415. String roomId = musicScoreData.getRoomId();
  1416. Long courseScheduleId = Long.parseLong(roomId.substring(1));
  1417. List<CourseScheduleStudentMusicScore> scheduleStudentMusicScores =
  1418. courseScheduleStudentMusicScoreDao.queryByScoreIdAndCourseId(musicScoreData.getMusicScoreAccompanimentId(),
  1419. courseScheduleId,null,null,0);
  1420. SysMusicScoreAccompaniment accompaniment = sysMusicScoreAccompanimentDao.get(musicScoreData.getMusicScoreAccompanimentId());
  1421. if(scheduleStudentMusicScores.size() == 0){
  1422. //第一次下载,生成数据
  1423. List<CourseScheduleStudentPayment> courseScheduleStudentPayments = courseScheduleStudentPaymentDao.findByCourseSchedule(courseScheduleId);
  1424. Set<Integer> studentIds = courseScheduleStudentPayments.stream().map(e -> e.getUserId()).collect(Collectors.toSet());
  1425. studentIds.forEach(e->{
  1426. CourseScheduleStudentMusicScore musicScore = new CourseScheduleStudentMusicScore();
  1427. musicScore.setMusicScoreAccompanimentId(accompaniment.getId());
  1428. musicScore.setSpeed(accompaniment.getSpeed());
  1429. musicScore.setCourseScheduleId(courseScheduleId);
  1430. musicScore.setUserId(e);
  1431. scheduleStudentMusicScores.add(musicScore);
  1432. });
  1433. CourseScheduleStudentMusicScore musicScore = new CourseScheduleStudentMusicScore();
  1434. musicScore.setMusicScoreAccompanimentId(accompaniment.getId());
  1435. musicScore.setSpeed(accompaniment.getSpeed());
  1436. musicScore.setCourseScheduleId(courseScheduleId);
  1437. musicScore.setUserId(authUser.getId());
  1438. musicScore.setUserType(1);
  1439. scheduleStudentMusicScores.add(musicScore);
  1440. courseScheduleStudentMusicScoreDao.batchInsert(scheduleStudentMusicScores);
  1441. }
  1442. MusicScoreMessage musicScoreMessage = JSON.parseObject(JSON.toJSONString(accompaniment), MusicScoreMessage.class);
  1443. MusicScoreDownloadMessageMessage msg = new MusicScoreDownloadMessageMessage(musicScoreMessage);
  1444. imHelper.publishMessage(authUser.getId().toString(), roomId, msg, 0);
  1445. }
  1446. @Override
  1447. @Transactional(rollbackFor = Exception.class)
  1448. public void pushDownloadExamSongMsg(String roomId, Integer examSongId) throws Exception {
  1449. SysUser authUser = sysUserFeignService.queryUserInfo();
  1450. SysExamSong sysExamSong = sysExamSongDao.get(examSongId);
  1451. if(sysExamSong == null){
  1452. throw new BizException("曲目信息不存在");
  1453. }
  1454. //学员曲目下载状态改为未下载
  1455. ExamSongDownloadData json = new ExamSongDownloadData();
  1456. json.setExamSongName(sysExamSong.getName());
  1457. json.setUrl(sysExamSong.getUrl());
  1458. json.setStatus(0);
  1459. json.setExamSongId(examSongId);
  1460. courseScheduleStudentPaymentDao.adjustExamSong(Long.parseLong(roomId.substring(1)),null, JSON.toJSONString(json));
  1461. ExamSongMessage examSongMessage = new ExamSongMessage();
  1462. examSongMessage.setExamSongName(sysExamSong.getName());
  1463. examSongMessage.setUrl(sysExamSong.getUrl());
  1464. examSongMessage.setExamSongId(examSongId);
  1465. ExamSongDownloadMessageMessage msg = new ExamSongDownloadMessageMessage(examSongMessage);
  1466. imHelper.publishMessage(authUser.getId().toString(), roomId, msg, 0);
  1467. }
  1468. @Override
  1469. public List<RongyunBasicUserDto> queryNoJoinStu(String roomId) {
  1470. return courseScheduleStudentPaymentDao.queryNoJoinStu(roomId,roomId.substring(1));
  1471. }
  1472. @Override
  1473. @Transactional(rollbackFor = Exception.class)
  1474. public void adjustMusicScore(MusicScoreData musicScoreData) throws Exception {
  1475. SysUser authUser = sysUserFeignService.queryUserInfo();
  1476. Integer studentId = authUser.getId();
  1477. String roomId = musicScoreData.getRoomId();
  1478. Long scheduleId = Long.parseLong(roomId.substring(1));
  1479. Integer accompanimentId = musicScoreData.getMusicScoreAccompanimentId();
  1480. List<CourseScheduleStudentMusicScore> studentMusicScores = courseScheduleStudentMusicScoreDao.queryByScoreIdAndCourseId(accompanimentId,scheduleId,studentId,null,null);
  1481. if(accompanimentId != null){
  1482. SysMusicScoreAccompaniment accompaniment = sysMusicScoreAccompanimentDao.get(accompanimentId);
  1483. if(accompaniment == null){
  1484. throw new BizException("曲目信息不存在");
  1485. }
  1486. //修改下载状态
  1487. if(studentMusicScores == null || studentMusicScores.size() == 0){
  1488. throw new BizException("学员不存在此下载曲目");
  1489. }
  1490. CourseScheduleStudentMusicScore studentMusicScore = studentMusicScores.get(0);
  1491. studentMusicScore.setDownStatus(musicScoreData.getStatus());
  1492. courseScheduleStudentMusicScoreDao.update(studentMusicScore);
  1493. }
  1494. //给老师发送学员曲目下载状态
  1495. CourseSchedule courseSchedule = courseScheduleDao.get(scheduleId);
  1496. MusicScoreDownloadStatusMessage statusMessage = new MusicScoreDownloadStatusMessage(studentId,studentMusicScores);
  1497. imHelper.publishMessage(studentId.toString(),courseSchedule.getActualTeacherId().toString(), roomId,statusMessage);
  1498. }
  1499. @Override
  1500. @Transactional(rollbackFor = Exception.class)
  1501. public void adjustExamSong(String roomId, Integer status,Integer examSongId) throws Exception {
  1502. if(roomId == null || status == null || examSongId == null){
  1503. throw new BizException("参数校验失败");
  1504. }
  1505. SysUser authUser = sysUserFeignService.queryUserInfo();
  1506. long scheduleId = Long.parseLong(roomId.substring(1));
  1507. SysExamSong sysExamSong = sysExamSongDao.get(examSongId);
  1508. if(sysExamSong == null){
  1509. throw new BizException("曲目信息不存在");
  1510. }
  1511. ExamSongDownloadData msg;
  1512. String examJson = courseScheduleStudentPaymentDao.getExamJsonByCourseIdAndUserId(scheduleId, authUser.getId());
  1513. if(StringUtils.isEmpty(examJson)){
  1514. msg = new ExamSongDownloadData();
  1515. msg.setExamSongName(sysExamSong.getName());
  1516. msg.setUrl(sysExamSong.getUrl());
  1517. msg.setStatus(status);
  1518. msg.setExamSongId(examSongId);
  1519. }else {
  1520. msg = JSON.parseObject(examJson, ExamSongDownloadData.class);
  1521. msg.setStatus(status);
  1522. }
  1523. courseScheduleStudentPaymentDao.adjustExamSong(scheduleId,authUser.getId(),JSON.toJSONString(msg));
  1524. //给老师发送学员曲目下载状态
  1525. CourseSchedule courseSchedule = courseScheduleDao.get(scheduleId);
  1526. ExamSongDownloadStatusMessage deviceResourceMessage = new ExamSongDownloadStatusMessage(status,authUser.getId(),examSongId);
  1527. imHelper.publishMessage(authUser.getId().toString(),courseSchedule.getActualTeacherId().toString(), roomId, deviceResourceMessage);
  1528. }
  1529. public void updateDisplay(String roomId, String senderId, String display, Integer isIncludeSender) throws Exception {
  1530. roomDao.updateDisplayByRid(roomId, display);
  1531. DisplayMessage displayMessage = new DisplayMessage(display);
  1532. imHelper.publishMessage(senderId, roomId, displayMessage, isIncludeSender);
  1533. }
  1534. public boolean isUserDisplay(Room room, String userId) {
  1535. boolean result = false;
  1536. if (!room.getDisplay().isEmpty() && room.getDisplay().contains("userId=" + userId)) {
  1537. if (room.getDisplay().contains("type=0") || room.getDisplay().contains("type=1") || room.getDisplay().contains("type=3")) {
  1538. result = true;
  1539. }
  1540. }
  1541. return result;
  1542. }
  1543. }