|  |  |  | 
|---|
|  |  |  | import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; | 
|---|
|  |  |  | import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper; | 
|---|
|  |  |  | import com.doumee.biz.system.SystemDictDataBiz; | 
|---|
|  |  |  | import com.doumee.core.erp.model.openapi.request.erp.UserFailRequest; | 
|---|
|  |  |  | import com.doumee.core.constants.ResponseStatus; | 
|---|
|  |  |  | import com.doumee.core.exception.BusinessException; | 
|---|
|  |  |  | import com.doumee.core.haikang.model.HKConstants; | 
|---|
|  |  |  | import com.doumee.core.haikang.model.param.BaseListPageResponse; | 
|---|
|  |  |  | import com.doumee.core.haikang.model.param.BaseResponse; | 
|---|
|  |  |  | import com.doumee.core.haikang.model.param.request.*; | 
|---|
|  |  |  | import com.doumee.core.haikang.model.param.request.event.visit.EventVisitInfoRequest; | 
|---|
|  |  |  | import com.doumee.core.haikang.model.param.respose.*; | 
|---|
|  |  |  | import com.doumee.core.haikang.service.HKService; | 
|---|
|  |  |  | import com.doumee.core.utils.Constants; | 
|---|
|  |  |  | import com.doumee.core.utils.DESUtil; | 
|---|
|  |  |  | import com.doumee.core.utils.DateUtil; | 
|---|
|  |  |  | import com.doumee.core.utils.ImageBase64Util; | 
|---|
|  |  |  | import com.doumee.dao.business.CompanyMapper; | 
|---|
|  |  |  | import com.doumee.core.wx.wxPlat.WxPlatNotice; | 
|---|
|  |  |  | import com.doumee.dao.business.DeviceRoleMapper; | 
|---|
|  |  |  | import com.doumee.dao.business.MemberMapper; | 
|---|
|  |  |  | import com.doumee.dao.business.VisitsMapper; | 
|---|
|  |  |  | import com.doumee.dao.business.RetentionMapper; | 
|---|
|  |  |  | import com.doumee.dao.business.join.VisitsJoinMapper; | 
|---|
|  |  |  | import com.doumee.dao.business.model.Company; | 
|---|
|  |  |  | import com.doumee.dao.business.model.DeviceRole; | 
|---|
|  |  |  | import com.doumee.dao.business.model.Member; | 
|---|
|  |  |  | import com.doumee.dao.business.model.Retention; | 
|---|
|  |  |  | import com.doumee.dao.business.model.Visits; | 
|---|
|  |  |  | import com.doumee.service.business.ERPSyncService; | 
|---|
|  |  |  | import com.github.yulichang.wrapper.MPJLambdaWrapper; | 
|---|
|  |  |  | import lombok.extern.slf4j.Slf4j; | 
|---|
|  |  |  | import org.apache.commons.lang3.StringUtils; | 
|---|
|  |  |  | import org.springframework.beans.factory.annotation.Autowired; | 
|---|
|  |  |  | import org.springframework.stereotype.Service; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import java.util.ArrayList; | 
|---|
|  |  |  | import java.util.Date; | 
|---|
|  |  |  | import java.util.List; | 
|---|
|  |  |  | import java.util.UUID; | 
|---|
|  |  |  | import java.util.*; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 海康访客业务Service实现 | 
|---|
|  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private VisitsJoinMapper visitsMapper; | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private RetentionMapper retentionMapper; | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private SystemDictDataBiz systemDictDataBiz; | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private DeviceRoleMapper deviceRoleMapper; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private WxPlatNotice wxPlatNotice; | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 同步海康组织信息,根据erp同步组织结果,定时检查需要下发到海康的组织信息 | 
|---|
|  |  |  | * 同步海康访客信息,根据erp同步组织结果,定时检查需要下发到海康的组织信息 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public void getOutTimeVisitRecord(){ | 
|---|
|  |  |  | if(Constants.DEALING_HK_VISIT_EXPIRE){ | 
|---|
|  |  |  | return   ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Constants.DEALING_HK_VISIT_EXPIRE =true; | 
|---|
|  |  |  | try { | 
|---|
|  |  |  | //查询所有需要同步的数据 | 
|---|
|  |  |  | List<Visits> list = getExpireVisitList(); | 
|---|
|  |  |  | //按照父级申请分批处理每次申请数据 | 
|---|
|  |  |  | if(list ==null || list.size()==0){ | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Date date = new Date(); | 
|---|
|  |  |  | for(Visits c : list) { | 
|---|
|  |  |  | //发起海康预约接口(需要登记),根据预约返回接口封装申请记录更新字段 | 
|---|
|  |  |  | AppointmentInfoResponse model =  getVisitRecord(c.getHkId()); | 
|---|
|  |  |  | if(model == null){ | 
|---|
|  |  |  | Visits update = new Visits(); | 
|---|
|  |  |  | //已失效 | 
|---|
|  |  |  | update.setStatus(Constants.VisitStatus.invalid); | 
|---|
|  |  |  | update.setEditDate(date); | 
|---|
|  |  |  | update.setId(c.getId()); | 
|---|
|  |  |  | update.setRemark("预约已被清除"); | 
|---|
|  |  |  | visitsMapper.updateById(update); | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | *      * 0:待审核(预约待审批)、 | 
|---|
|  |  |  | *      * 1:正常(预约成功或预约审批成功后<未登记>:当前时间未超过预计来访时间)、 | 
|---|
|  |  |  | *      * 2:迟到(预约成功或预约审批成功后<未登记>:当前时间超过预计来访时间,但未超过预计离开时间)、 | 
|---|
|  |  |  | *      * 3:失效(预约成功或预约审批成功后<未登记>:当前时间已超过预计离开时间)、 | 
|---|
|  |  |  | *      * 4:审核退回(预约待审批,审批人员审批退回)、 | 
|---|
|  |  |  | *      * 5:超期自动签离(超期未签离记录访客,由后台任务定时处理签离)、 | 
|---|
|  |  |  | *      * 6:已签离(访客签离)、 | 
|---|
|  |  |  | *      * 7:超期未签离(访客登记,当前时间已超过预计离开时间,还未进行签离)、 | 
|---|
|  |  |  | *      * 8:已到达;(访客登记,当前时间未超过预计离开时间)、 | 
|---|
|  |  |  | *      * 9:审核失效(预约待审批,一直到当前时间超过预计离开时间,还未审批通过)、 | 
|---|
|  |  |  | *      * 10:邀约中(员工发起邀约,访客还未应邀)、 | 
|---|
|  |  |  | *      * 11:邀约失效(员工发起邀约,一直到当前时间超过预计离开时间,访客还未应邀) | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | if (model.getVisitorStatus()!=null && ",3,4,11,".contains( ","  +model.getVisitorStatus()+"," )){ | 
|---|
|  |  |  | //对相应状态下的数据进行【已失效】处理 | 
|---|
|  |  |  | Visits update = new Visits(); | 
|---|
|  |  |  | //已失效 | 
|---|
|  |  |  | update.setStatus(Constants.VisitStatus.invalid); | 
|---|
|  |  |  | update.setEditDate(date); | 
|---|
|  |  |  | update.setId(c.getId()); | 
|---|
|  |  |  | update.setInDate(DateUtil.getISO8601DateByStr2(model.getVisitStartTime())); | 
|---|
|  |  |  | update.setOutDate(DateUtil.getISO8601DateByStr2(model.getVisitEndTime())); | 
|---|
|  |  |  | update.setRemark("超时未登记"); | 
|---|
|  |  |  | visitsMapper.updateById(update); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (model.getVisitorStatus()!=null && ",5,6,".contains( ","  +model.getVisitorStatus()+"," )){ | 
|---|
|  |  |  | // 对相应状态下的数据进行【已签离】处理 | 
|---|
|  |  |  | if(!Constants.equalsInteger(c.getStatus(),Constants.VisitStatus.signout)){ | 
|---|
|  |  |  | Visits update = new Visits(); | 
|---|
|  |  |  | //已失效 | 
|---|
|  |  |  | update.setStatus(Constants.VisitStatus.invalid); | 
|---|
|  |  |  | update.setEditDate(date); | 
|---|
|  |  |  | update.setId(c.getId()); | 
|---|
|  |  |  | update.setInDate(DateUtil.getISO8601DateByStr2(model.getVisitStartTime())); | 
|---|
|  |  |  | update.setOutDate(DateUtil.getISO8601DateByStr2(model.getVisitEndTime())); | 
|---|
|  |  |  | update.setRemark("已签离"); | 
|---|
|  |  |  | visitsMapper.updateById(update); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //先删除原有的在场人员(普通访客) | 
|---|
|  |  |  | retentionMapper.delete(new UpdateWrapper<Retention>().lambda() | 
|---|
|  |  |  | .eq(Retention::getType,Constants.memberType.visitor) | 
|---|
|  |  |  | .eq(Retention::getMemberId,c.getMemberId())); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (model.getVisitorStatus()!=null&& ",7,8,".contains( ","  +model.getVisitorStatus()+"," )){ | 
|---|
|  |  |  | //如果已登记 | 
|---|
|  |  |  | if(!Constants.equalsInteger(c.getStatus(),Constants.VisitStatus.signin)){ | 
|---|
|  |  |  | Visits update = new Visits(); | 
|---|
|  |  |  | //已失效 | 
|---|
|  |  |  | update.setStatus(Constants.VisitStatus.signin); | 
|---|
|  |  |  | update.setEditDate(date); | 
|---|
|  |  |  | update.setId(c.getId()); | 
|---|
|  |  |  | update.setInDate(DateUtil.getISO8601DateByStr2(model.getVisitStartTime())); | 
|---|
|  |  |  | update.setOutDate(DateUtil.getISO8601DateByStr2(model.getVisitEndTime())); | 
|---|
|  |  |  | update.setRemark("超时未签到"); | 
|---|
|  |  |  | visitsMapper.updateById(update); | 
|---|
|  |  |  | //先删除原有的在场人员(普通访客) | 
|---|
|  |  |  | retentionMapper.delete(new UpdateWrapper<Retention>().lambda() | 
|---|
|  |  |  | .eq(Retention::getType,Constants.memberType.visitor) | 
|---|
|  |  |  | .eq(Retention::getMemberId,c.getMemberId())); | 
|---|
|  |  |  | //再插入最新的在厂人员 | 
|---|
|  |  |  | retentionMapper.insert(getRetentionModelByVisitRequest(c,update.getInDate())); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }catch (Exception e){ | 
|---|
|  |  |  | e.printStackTrace(); | 
|---|
|  |  |  | }finally { | 
|---|
|  |  |  | Constants.DEALING_HK_VISIT_EXPIRE =false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private Retention getRetentionModelByVisitRequest(Visits visits,Date date) { | 
|---|
|  |  |  | Retention retention = new Retention(); | 
|---|
|  |  |  | retention.setIsdeleted(Constants.ZERO); | 
|---|
|  |  |  | retention.setCreateDate(date); | 
|---|
|  |  |  | retention.setClasses(visits.getClasses()); | 
|---|
|  |  |  | retention.setCode(visits.getCode()); | 
|---|
|  |  |  | retention.setIdcardNo(visits.getIdcardNo()); | 
|---|
|  |  |  | retention.setIdcardDecode(visits.getIdcardDecode()); | 
|---|
|  |  |  | retention.setName(visits.getName()); | 
|---|
|  |  |  | retention.setBirthday(visits.getBirthday()); | 
|---|
|  |  |  | retention.setType(visits.getMemberType()); | 
|---|
|  |  |  | retention.setCompanyId(visits.getCompanyId()); | 
|---|
|  |  |  | retention.setCompanyName(visits.getCompanyName()); | 
|---|
|  |  |  | retention.setEventDate(retention.getCreateDate()); | 
|---|
|  |  |  | retention.setFaceImg(visits.getFaceImg()); | 
|---|
|  |  |  | retention.setImgurl(visits.getImgurl()); | 
|---|
|  |  |  | retention.setPhone(visits.getPhone()); | 
|---|
|  |  |  | retention.setMemberId(visits.getMemberId()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return retention; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public AppointmentInfoResponse getVisitRecord(String orderId){ | 
|---|
|  |  |  | //分页遍历循环查询所有门禁设备数据 | 
|---|
|  |  |  | if(StringUtils.isBlank(orderId)){ | 
|---|
|  |  |  | return  null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | AppointmentListRequest param =  new AppointmentListRequest(); | 
|---|
|  |  |  | param.setPageNo(1); | 
|---|
|  |  |  | param.setPageSize(1); | 
|---|
|  |  |  | param.setOrderId(orderId); | 
|---|
|  |  |  | BaseResponse<BaseListPageResponse<AppointmentInfoResponse>> response = HKService.appointmentRecords(param); | 
|---|
|  |  |  | if(response == null || !StringUtils.equals(response.getCode(), HKConstants.RESPONSE_SUCCEE)  ){ | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if(response.getData() == null || response.getData().getList() == null|| response.getData().getList().size() ==0){ | 
|---|
|  |  |  | return  null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return response.getData().getList().get(0); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public  void syncVisitData(){ | 
|---|
|  |  |  | if(Constants.DEALING_HK_VISIT){ | 
|---|
|  |  |  | 
|---|
|  |  |  | String type = Constants.equalsInteger(c.getType(),Constants.ONE)?applyType:applyLwType; | 
|---|
|  |  |  | getUpdateModelByResponse(c,date,type,roleList,path); | 
|---|
|  |  |  | visitsMapper.updateById(c); | 
|---|
|  |  |  | if(Objects.isNull(c.getParentId())){ | 
|---|
|  |  |  | wxPlatNotice.sendVisitAuditTemplateNotice(c, | 
|---|
|  |  |  | systemDictDataBiz.queryByCode(Constants.WX_PLATFORM,Constants.WX_PLATFORM_PREFIX).getCode(), | 
|---|
|  |  |  | systemDictDataBiz.queryByCode(Constants.WX_PLATFORM,Constants.WX_PLATFORM_AUDIT_VISIT).getCode()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }catch (Exception e){ | 
|---|
|  |  |  | e.printStackTrace(); | 
|---|
|  |  |  | 
|---|
|  |  |  | private void getUpdateModelByResponse(  Visits c,Date date,String type,List<DeviceRole> roleList,String path ) { | 
|---|
|  |  |  | String code = null; | 
|---|
|  |  |  | String id = null; | 
|---|
|  |  |  | String qrcode = null; | 
|---|
|  |  |  | //发起海康预约接口 | 
|---|
|  |  |  | if(StringUtils.equals(type,"0")){ | 
|---|
|  |  |  | //(需要登记) | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | BaseResponse<VisitAppointmentResponse> response =  HKService.visitAppiontment(request); | 
|---|
|  |  |  | code =response!=null ?response.getCode():null; | 
|---|
|  |  |  | id = (response!=null && response.getData()!=null | 
|---|
|  |  |  | if((response!=null && response.getData()!=null | 
|---|
|  |  |  | && response.getData().getAppointmentInfoList() !=null | 
|---|
|  |  |  | && response.getData().getAppointmentInfoList().size()>0)?response.getData().getAppointmentInfoList().get(0).getOrderId():null; | 
|---|
|  |  |  | && response.getData().getAppointmentInfoList().size()>0)){ | 
|---|
|  |  |  | id =response.getData().getAppointmentInfoList().get(0).getOrderId(); | 
|---|
|  |  |  | qrcode =response.getData().getAppointmentInfoList().get(0).getQRCode(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | //免登记 | 
|---|
|  |  |  | VisitAppointmentMDJRequest request =getHkMDJRequestPara(c,roleList,path); | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | BaseResponse<VisitAppointmentMDJResponse> response = HKService.visitAppiontmentMDJ(request ); | 
|---|
|  |  |  | code =response!=null ?response.getCode():null; | 
|---|
|  |  |  | id = (response!=null && response.getData()!=null)?response.getData().getOrderId():null; | 
|---|
|  |  |  | if(response!=null && response.getData()!=null){ | 
|---|
|  |  |  | id = response.getData().getOrderId(); | 
|---|
|  |  |  | qrcode =response.getData().getQRCode(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (code!= null && id!=null) { | 
|---|
|  |  |  | //海康下发成功 | 
|---|
|  |  |  | c.setHkId( id);//预约标识 | 
|---|
|  |  |  | c.setQrcode(qrcode); | 
|---|
|  |  |  | c.setRemark("下发海康成功!【"+type+"】"); | 
|---|
|  |  |  | c.setStatus(Constants.VisitStatus.xfSuccess);//下发海康成功 | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | //海康下发成功 | 
|---|
|  |  |  | //海康下发失败 | 
|---|
|  |  |  | c.setRemark("下发海康失败!【"+type+"】"); | 
|---|
|  |  |  | c.setStatus(Constants.VisitStatus.xfFail);//下发海康失败 | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | //人脸为空,不能进行推送 | 
|---|
|  |  |  | return  null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | request.setVisitStartTime(DateUtil.getISO8601Timestamp(c.getStarttime())); | 
|---|
|  |  |  | request.setVisitEndTime(DateUtil.getISO8601Timestamp(c.getEndtime())); | 
|---|
|  |  |  | request.setVisitStartTime(DateUtil.getISO8601Timestamp2(c.getStarttime())); | 
|---|
|  |  |  | request.setVisitEndTime(DateUtil.getISO8601Timestamp2(c.getEndtime())); | 
|---|
|  |  |  | request.setReceptionistId(c.getReceptMemberHkId());//被访人海康编码 | 
|---|
|  |  |  | request.setVisitPurpose(c.getReason()); | 
|---|
|  |  |  | request.setVisitorInfo(info); | 
|---|
|  |  |  | 
|---|
|  |  |  | info.setVisitorWorkUnit(c.getCompanyName()); | 
|---|
|  |  |  | info.setPhoneNo(c.getPhone()); | 
|---|
|  |  |  | info.setPlateNo(c.getCarNos()); | 
|---|
|  |  |  | info.setGender(c.getSex()+""); | 
|---|
|  |  |  | if(Constants.equalsInteger(c.getIdcardType(),Constants.ZERO)){ | 
|---|
|  |  |  | info.setCertificateType(HKConstants.CertificateType.SHENFENZHENG.getKey()+""); | 
|---|
|  |  |  | info.setCertificateNo(DESUtil.decrypt(Constants.EDS_PWD, c.getIdcardNo())); | 
|---|
|  |  |  | 
|---|
|  |  |  | MPJLambdaWrapper<Visits> queryWrapper = new MPJLambdaWrapper<>(); | 
|---|
|  |  |  | queryWrapper.selectAll(Visits.class); | 
|---|
|  |  |  | queryWrapper.selectAs(Member::getName,Visits::getReceptMemberName) | 
|---|
|  |  |  | .select("t1.hk_id as receptMemberHkId") | 
|---|
|  |  |  | .selectAs(Member::getHkId,Visits::getReceptMemberHkId) | 
|---|
|  |  |  | .leftJoin(Member.class,Member::getId,Visits::getReceptMemberId); | 
|---|
|  |  |  | queryWrapper.eq(Visits::getStatus,Constants.VisitStatus.pass)//审批通过 | 
|---|
|  |  |  | .eq(Visits::getHkStatus,Constants.ZERO)//未同步到海康 | 
|---|
|  |  |  | 
|---|
|  |  |  | List<Visits> list = visitsMapper.selectJoinList(Visits.class,queryWrapper); | 
|---|
|  |  |  | return list; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | private List<Visits> getExpireVisitList() { | 
|---|
|  |  |  | MPJLambdaWrapper<Visits> queryWrapper = new MPJLambdaWrapper<>(); | 
|---|
|  |  |  | queryWrapper.selectAll(Visits.class) | 
|---|
|  |  |  | .selectAs(Member::getType,Visits::getMemberType) | 
|---|
|  |  |  | .leftJoin(Member.class,Member::getId,Visits::getMemberId); | 
|---|
|  |  |  | queryWrapper.in(Visits::getStatus, Arrays.asList(new Integer[]{Constants.VisitStatus.xfSuccess,Constants.VisitStatus.signin}) ); | 
|---|
|  |  |  | List<Visits> list = visitsMapper.selectJoinList(Visits.class,queryWrapper); | 
|---|
|  |  |  | return list; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|