mp:增加【自动回复】的后端新增、修改、删除接口

pull/2/head
YunaiV 2023-01-17 00:58:35 +08:00
parent 1e402d8063
commit 6fc0b3fc54
8 changed files with 210 additions and 32 deletions

View File

@ -55,4 +55,10 @@ public interface ErrorCodeConstants {
ErrorCode MENU_SAVE_FAIL = new ErrorCode(1006008000, "创建菜单失败,原因:{}"); ErrorCode MENU_SAVE_FAIL = new ErrorCode(1006008000, "创建菜单失败,原因:{}");
ErrorCode MENU_DELETE_FAIL = new ErrorCode(1006008001, "删除菜单失败,原因:{}"); ErrorCode MENU_DELETE_FAIL = new ErrorCode(1006008001, "删除菜单失败,原因:{}");
// ========== 公众号自动回复 1006009000============
ErrorCode AUTO_REPLY_NOT_EXISTS = new ErrorCode(1006009000, "自动回复不存在");
ErrorCode AUTO_REPLY_ADD_SUBSCRIBE_FAIL_EXISTS = new ErrorCode(1006009001, "操作失败,原因:已存在关注时的回复");
ErrorCode AUTO_REPLY_ADD_MESSAGE_FAIL_EXISTS = new ErrorCode(1006009002, "操作失败,原因:已存在该消息类型的回复");
ErrorCode AUTO_REPLY_ADD_KEYWORD_FAIL_EXISTS = new ErrorCode(1006009003, "操作失败,原因:已关在该关键字的回复");
} }

View File

@ -2,7 +2,9 @@ package cn.iocoder.yudao.module.mp.controller.admin.message;
import cn.iocoder.yudao.framework.common.pojo.CommonResult; import cn.iocoder.yudao.framework.common.pojo.CommonResult;
import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.pojo.PageResult;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyCreateReqVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyRespVO; import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyRespVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyUpdateReqVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.message.MpMessagePageReqVO; import cn.iocoder.yudao.module.mp.controller.admin.message.vo.message.MpMessagePageReqVO;
import cn.iocoder.yudao.module.mp.convert.message.MpAutoReplyConvert; import cn.iocoder.yudao.module.mp.convert.message.MpAutoReplyConvert;
import cn.iocoder.yudao.module.mp.dal.dataobject.message.MpAutoReplyDO; import cn.iocoder.yudao.module.mp.dal.dataobject.message.MpAutoReplyDO;
@ -12,10 +14,7 @@ import io.swagger.annotations.ApiImplicitParam;
import io.swagger.annotations.ApiOperation; import io.swagger.annotations.ApiOperation;
import org.springframework.security.access.prepost.PreAuthorize; import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.validation.annotation.Validated; import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.*;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import javax.annotation.Resource; import javax.annotation.Resource;
import javax.validation.Valid; import javax.validation.Valid;
@ -48,4 +47,28 @@ public class MpAutoReplyController {
return success(MpAutoReplyConvert.INSTANCE.convert(autoReply)); return success(MpAutoReplyConvert.INSTANCE.convert(autoReply));
} }
@PostMapping("/create")
@ApiOperation("创建公众号自动回复")
@PreAuthorize("@ss.hasPermission('mp:auto-reply:create')")
public CommonResult<Long> createAutoReply(@Valid @RequestBody MpAutoReplyCreateReqVO createReqVO) {
return success(mpAutoReplyService.createAutoReply(createReqVO));
}
@PutMapping("/update")
@ApiOperation("更新公众号自动回复")
@PreAuthorize("@ss.hasPermission('mp:auto-reply:update')")
public CommonResult<Boolean> updateAutoReply(@Valid @RequestBody MpAutoReplyUpdateReqVO updateReqVO) {
mpAutoReplyService.updateAutoReply(updateReqVO);
return success(true);
}
@DeleteMapping("/delete")
@ApiOperation("删除公众号自动回复")
@ApiImplicitParam(name = "id", value = "编号", required = true, dataTypeClass = Long.class)
@PreAuthorize("@ss.hasPermission('mp:auto-reply:delete')")
public CommonResult<Boolean> deleteAutoReply(@RequestParam("id") Long id) {
mpAutoReplyService.deleteAutoReply(id);
return success(true);
}
} }

View File

@ -18,8 +18,4 @@ public class MpAutoReplyCreateReqVO extends MpAutoReplyBaseVO {
@NotNull(message = "微信公众号 ID不能为空") @NotNull(message = "微信公众号 ID不能为空")
private Long accountId; private Long accountId;
@ApiModelProperty(value = "回复类型", required = true, example = "1", notes = "参见 MpAutoReplyTypeEnum 枚举")
@NotNull(message = "回复类型不能为空")
private Integer type;
} }

View File

@ -1,7 +1,9 @@
package cn.iocoder.yudao.module.mp.convert.message; package cn.iocoder.yudao.module.mp.convert.message;
import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.pojo.PageResult;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyCreateReqVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyRespVO; import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyRespVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyUpdateReqVO;
import cn.iocoder.yudao.module.mp.dal.dataobject.message.MpAutoReplyDO; import cn.iocoder.yudao.module.mp.dal.dataobject.message.MpAutoReplyDO;
import cn.iocoder.yudao.module.mp.service.message.bo.MpMessageSendOutReqBO; import cn.iocoder.yudao.module.mp.service.message.bo.MpMessageSendOutReqBO;
import org.mapstruct.Mapper; import org.mapstruct.Mapper;
@ -29,4 +31,7 @@ public interface MpAutoReplyConvert {
MpAutoReplyRespVO convert(MpAutoReplyDO bean); MpAutoReplyRespVO convert(MpAutoReplyDO bean);
MpAutoReplyDO convert(MpAutoReplyCreateReqVO bean);
MpAutoReplyDO convert(MpAutoReplyUpdateReqVO bean);
} }

View File

@ -49,4 +49,22 @@ public interface MpAutoReplyMapper extends BaseMapperX<MpAutoReplyDO> {
.eq(MpAutoReplyDO::getType, MpAutoReplyTypeEnum.SUBSCRIBE.getType())); .eq(MpAutoReplyDO::getType, MpAutoReplyTypeEnum.SUBSCRIBE.getType()));
} }
default MpAutoReplyDO selectByAccountIdAndSubscribe(Long accountId) {
return selectOne(MpAutoReplyDO::getAccountId, accountId,
MpAutoReplyDO::getType, MpAutoReplyTypeEnum.SUBSCRIBE.getType());
}
default MpAutoReplyDO selectByAccountIdAndMessage(Long accountId, String requestMessageType) {
return selectOne(new LambdaQueryWrapperX<MpAutoReplyDO>()
.eq(MpAutoReplyDO::getAccountId, accountId)
.eq(MpAutoReplyDO::getType, MpAutoReplyTypeEnum.MESSAGE.getType())
.eq(MpAutoReplyDO::getRequestMessageType, requestMessageType));
}
default MpAutoReplyDO selectByAccountIdAndKeyword(Long accountId, String requestKeyword) {
return selectOne(new LambdaQueryWrapperX<MpAutoReplyDO>()
.eq(MpAutoReplyDO::getAccountId, accountId)
.eq(MpAutoReplyDO::getType, MpAutoReplyTypeEnum.KEYWORD.getType())
.eq(MpAutoReplyDO::getRequestKeyword, requestKeyword));
}
} }

View File

@ -1,6 +1,8 @@
package cn.iocoder.yudao.module.mp.service.message; package cn.iocoder.yudao.module.mp.service.message;
import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.pojo.PageResult;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyCreateReqVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyUpdateReqVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.message.MpMessagePageReqVO; import cn.iocoder.yudao.module.mp.controller.admin.message.vo.message.MpMessagePageReqVO;
import cn.iocoder.yudao.module.mp.dal.dataobject.message.MpAutoReplyDO; import cn.iocoder.yudao.module.mp.dal.dataobject.message.MpAutoReplyDO;
import me.chanjar.weixin.mp.bean.message.WxMpXmlMessage; import me.chanjar.weixin.mp.bean.message.WxMpXmlMessage;
@ -29,6 +31,29 @@ public interface MpAutoReplyService {
*/ */
MpAutoReplyDO getAutoReply(Long id); MpAutoReplyDO getAutoReply(Long id);
/**
*
*
* @param createReqVO
* @return
*/
Long createAutoReply(MpAutoReplyCreateReqVO createReqVO);
/**
*
*
* @param updateReqVO
*/
void updateAutoReply(MpAutoReplyUpdateReqVO updateReqVO);
/**
*
*
* @param id
*/
void deleteAutoReply(Long id);
/** /**
* *
* *
@ -46,4 +71,5 @@ public interface MpAutoReplyService {
* @return * @return
*/ */
WxMpXmlOutMessage replyForSubscribe(String appId, WxMpXmlMessage wxMessage); WxMpXmlOutMessage replyForSubscribe(String appId, WxMpXmlMessage wxMessage);
} }

View File

@ -2,13 +2,18 @@ package cn.iocoder.yudao.module.mp.service.message;
import cn.hutool.core.collection.CollUtil; import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.lang.Assert; import cn.hutool.core.lang.Assert;
import cn.hutool.core.util.ObjUtil;
import cn.iocoder.yudao.framework.common.exception.ErrorCode;
import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.pojo.PageResult;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyCreateReqVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.autoreply.MpAutoReplyUpdateReqVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.message.MpMessagePageReqVO; import cn.iocoder.yudao.module.mp.controller.admin.message.vo.message.MpMessagePageReqVO;
import cn.iocoder.yudao.module.mp.convert.message.MpAutoReplyConvert; import cn.iocoder.yudao.module.mp.convert.message.MpAutoReplyConvert;
import cn.iocoder.yudao.module.mp.dal.dataobject.account.MpAccountDO; import cn.iocoder.yudao.module.mp.dal.dataobject.account.MpAccountDO;
import cn.iocoder.yudao.module.mp.dal.dataobject.message.MpAutoReplyDO; import cn.iocoder.yudao.module.mp.dal.dataobject.message.MpAutoReplyDO;
import cn.iocoder.yudao.module.mp.dal.mysql.message.MpAutoReplyMapper; import cn.iocoder.yudao.module.mp.dal.mysql.message.MpAutoReplyMapper;
import cn.iocoder.yudao.module.mp.enums.message.MpAutoReplyTypeEnum; import cn.iocoder.yudao.module.mp.enums.message.MpAutoReplyTypeEnum;
import cn.iocoder.yudao.module.mp.framework.mp.core.util.MpUtils;
import cn.iocoder.yudao.module.mp.service.account.MpAccountService; import cn.iocoder.yudao.module.mp.service.account.MpAccountService;
import cn.iocoder.yudao.module.mp.service.message.bo.MpMessageSendOutReqBO; import cn.iocoder.yudao.module.mp.service.message.bo.MpMessageSendOutReqBO;
import me.chanjar.weixin.common.api.WxConsts; import me.chanjar.weixin.common.api.WxConsts;
@ -19,8 +24,12 @@ import org.springframework.stereotype.Service;
import org.springframework.validation.annotation.Validated; import org.springframework.validation.annotation.Validated;
import javax.annotation.Resource; import javax.annotation.Resource;
import javax.validation.Validator;
import java.util.List; import java.util.List;
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
import static cn.iocoder.yudao.module.mp.enums.ErrorCodeConstants.*;
/** /**
* Service * Service
* *
@ -36,6 +45,9 @@ public class MpAutoReplyServiceImpl implements MpAutoReplyService {
@Lazy // 延迟加载,避免循环依赖 @Lazy // 延迟加载,避免循环依赖
private MpAccountService mpAccountService; private MpAccountService mpAccountService;
@Resource
private Validator validator;
@Resource @Resource
private MpAutoReplyMapper mpAutoReplyMapper; private MpAutoReplyMapper mpAutoReplyMapper;
@ -49,6 +61,96 @@ public class MpAutoReplyServiceImpl implements MpAutoReplyService {
return mpAutoReplyMapper.selectById(id); return mpAutoReplyMapper.selectById(id);
} }
@Override
public Long createAutoReply(MpAutoReplyCreateReqVO createReqVO) {
// 第一步,校验数据
if (createReqVO.getResponseMessageType() != null) {
MpUtils.validateMessage(validator, createReqVO.getResponseMessageType(), createReqVO);
}
validateAutoReplyConflict(null, createReqVO.getAccountId(), createReqVO.getType(),
createReqVO.getRequestKeyword(), createReqVO.getRequestMessageType());
// 第二步,插入数据
MpAccountDO account = mpAccountService.getRequiredAccount(createReqVO.getAccountId());
MpAutoReplyDO autoReply = MpAutoReplyConvert.INSTANCE.convert(createReqVO)
.setAppId(account.getAppId());
mpAutoReplyMapper.insert(autoReply);
return autoReply.getId();
}
@Override
public void updateAutoReply(MpAutoReplyUpdateReqVO updateReqVO) {
// 第一步,校验数据
if (updateReqVO.getResponseMessageType() != null) {
MpUtils.validateMessage(validator, updateReqVO.getResponseMessageType(), updateReqVO);
}
MpAutoReplyDO autoReply = validateAutoReplyExists(updateReqVO.getId());
validateAutoReplyConflict(updateReqVO.getId(), autoReply.getAccountId(), updateReqVO.getType(),
updateReqVO.getRequestKeyword(), updateReqVO.getRequestMessageType());
// 第二步,更新数据
MpAutoReplyDO updateObj = MpAutoReplyConvert.INSTANCE.convert(updateReqVO)
.setAccountId(null).setAppId(null); // 避免前端传递,更新着两个字段
mpAutoReplyMapper.updateById(updateObj);
}
/**
*
*
* type
* 1. type = SUBSCRIBE
* 2. type = MESSAGE requestMessageType
* 3. type = KEYWORD keyword
*
* @param id
* @param accountId
* @param type
* @param requestKeyword
* @param requestMessageType
*/
private void validateAutoReplyConflict(Long id, Long accountId, Integer type,
String requestKeyword, String requestMessageType) {
// 获得已经存在的自动回复
MpAutoReplyDO autoReply = null;
ErrorCode errorCode = null;
if (MpAutoReplyTypeEnum.SUBSCRIBE.getType().equals(type)) {
autoReply = mpAutoReplyMapper.selectByAccountIdAndSubscribe(accountId);
errorCode = AUTO_REPLY_ADD_SUBSCRIBE_FAIL_EXISTS;
} else if (MpAutoReplyTypeEnum.MESSAGE.getType().equals(type)) {
autoReply = mpAutoReplyMapper.selectByAccountIdAndMessage(accountId, requestMessageType);
errorCode = AUTO_REPLY_ADD_MESSAGE_FAIL_EXISTS;
} else if (MpAutoReplyTypeEnum.KEYWORD.getType().equals(type)) {
autoReply = mpAutoReplyMapper.selectByAccountIdAndKeyword(accountId, requestKeyword);
errorCode = AUTO_REPLY_ADD_KEYWORD_FAIL_EXISTS;
}
if (autoReply == null) {
return;
}
// 存在冲突,抛出业务异常
if (id == null // 情况一新增id == null存在记录说明冲突
|| ObjUtil.notEqual(id, autoReply.getId())) { // 情况二修改id != nullid 不匹配,说明冲突
throw exception(errorCode);
}
}
@Override
public void deleteAutoReply(Long id) {
// 校验用户存在
validateAutoReplyExists(id);
// 删除自动回复
mpAutoReplyMapper.deleteById(id);
}
private MpAutoReplyDO validateAutoReplyExists(Long id) {
MpAutoReplyDO autoReply = mpAutoReplyMapper.selectById(id);
if (autoReply == null) {
throw exception(AUTO_REPLY_NOT_EXISTS);
}
return autoReply;
}
@Override @Override
public WxMpXmlOutMessage replyForMessage(String appId, WxMpXmlMessage wxMessage) { public WxMpXmlOutMessage replyForMessage(String appId, WxMpXmlMessage wxMessage) {
// 第一步,匹配自动回复 // 第一步,匹配自动回复

View File

@ -96,6 +96,10 @@ SOFTWARE.
</el-tab-pane> </el-tab-pane>
<el-tab-pane name="3"> <el-tab-pane name="3">
<span slot="label"><i class="el-icon-news"></i> 关键词回复</span> <span slot="label"><i class="el-icon-news"></i> 关键词回复</span>
</el-tab-pane>
</el-tabs>
<!-- 列表 -->
<el-table v-loading="loading" :data="list"> <el-table v-loading="loading" :data="list">
<el-table-column label="关键词" align="center" prop="requestKeyword"/> <el-table-column label="关键词" align="center" prop="requestKeyword"/>
<el-table-column label="匹配类型" align="center" prop="requestMatch"> <el-table-column label="匹配类型" align="center" prop="requestMatch">
@ -120,8 +124,6 @@ SOFTWARE.
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
</el-tab-pane>
</el-tabs>
<!-- 添加或修改自动回复的对话框 --> <!-- 添加或修改自动回复的对话框 -->
<el-dialog :title="title" :visible.sync="open" width="800px" append-to-body> <el-dialog :title="title" :visible.sync="open" width="800px" append-to-body>