临时提交:语音版

This commit is contained in:
墨大叔 2024-09-30 17:54:21 +08:00
parent 9d42395253
commit beb93dad38
26 changed files with 329 additions and 42 deletions

View File

@ -86,6 +86,11 @@ public class IotConstants {
public static final String COMMAND_SET_PASS = "pass";
/**
* 命令 设置语音播报阈值()
*/
public static final String COMMAND_SET_VOICE = "dj_set";
/**----------------------------命令end----------------------------*/
}

View File

@ -23,8 +23,6 @@ import java.util.Map;
@Service
public class TmPayService {
private final static String SIGNKEY = "b4ixpiogfj5vu3tbkv23gj0dvo2j2ksz";
@Autowired
private TmPayConfig config;
@ -147,7 +145,7 @@ public class TmPayService {
params.remove("sign");
// 按照请求时的签名逻辑生成签名字符串
String paramsStr = TmPayUtil.getAsciiSort(params); // 按ASCII排序
String generatedSign = TmPayUtil.getMD5Code(paramsStr + "&key=" + SIGNKEY).toUpperCase(); // 重新生成签名
String generatedSign = TmPayUtil.getMD5Code(paramsStr + "&key=" + config.getSignKey()).toUpperCase(); // 重新生成签名
System.out.println("新生成的签名-----------:"+generatedSign);
// 比较签名是否一致
return generatedSign.equals(receivedSign);

View File

@ -10,7 +10,7 @@ import lombok.Getter;
*/
@Getter
@AllArgsConstructor
public enum PayStatus {
public enum TmPayStatus {
SUCCESS("SUCCESS", "支付成功"),
NOTPAY("NOTPAY", "未支付"),
@ -36,6 +36,15 @@ public enum PayStatus {
return SUCCESS.getCode().equals(code);
}
/**
* 判断支付状态是否成功
* @param status 支付状态
* @return 是否成功
*/
public static boolean isSuccess(TmPayStatus status) {
return status != null && SUCCESS.getCode().equals(status.getCode());
}
/**
* 判断支付状态是否转入退款
* @param code 支付状态码
@ -50,8 +59,8 @@ public enum PayStatus {
* @param code 支付状态码
* @return 对应的PayStatus枚举
*/
public static PayStatus getByCode(String code) {
for (PayStatus status : PayStatus.values()) {
public static TmPayStatus getByCode(String code) {
for (TmPayStatus status : TmPayStatus.values()) {
if (status.getCode().equals(code)) {
return status;
}

View File

@ -10,7 +10,7 @@ import lombok.Getter;
*/
@Getter
@AllArgsConstructor
public enum RefundStatus {
public enum TmRefundStatus {
FAILED(0, "退款失败"),
REFUNDED(1, "转入退款");
@ -31,8 +31,8 @@ public enum RefundStatus {
* @param code 状态码
* @return 对应的 RefundStatus 枚举
*/
public static RefundStatus getByCode(int code) {
for (RefundStatus status : RefundStatus.values()) {
public static TmRefundStatus getByCode(int code) {
for (TmRefundStatus status : TmRefundStatus.values()) {
if (status.getCode() == code) {
return status;
}

View File

@ -1,6 +1,6 @@
package com.ruoyi.common.pay.tm.vo;
import com.ruoyi.common.pay.tm.enums.RefundStatus;
import com.ruoyi.common.pay.tm.enums.TmRefundStatus;
import lombok.Data;
/**
@ -23,7 +23,7 @@ public class RefundInfo {
private String fromType; // 订单来源: wx, alipay, web, mini, pos, pc, desktop, api
private String refundTime; // 退款成功时间
private String refundMessage; // 退款失败原因
private RefundStatus refundStatus; // 退款状态: RefundStatus 枚举
private TmRefundStatus refundStatus; // 退款状态: RefundStatus 枚举
private String refundAmount; // 已退款金额
private String createTime; // 创建时间
}

View File

@ -1,6 +1,6 @@
package com.ruoyi.common.pay.tm.vo;
import com.ruoyi.common.pay.tm.enums.PayStatus;
import com.ruoyi.common.pay.tm.enums.TmPayStatus;
import lombok.Data;
@Data
@ -132,7 +132,7 @@ public class TmTradeInfo {
* OPERATE_SETTLING押金消费已受理
* REVOKED_SUCCESS预授权请求撤销成功
*/
private PayStatus payStatus;
private TmPayStatus payStatus;
/**
* 支付时间

View File

@ -1,9 +1,6 @@
package com.ruoyi.iot.service;
import com.ruoyi.iot.domain.CurrentDeviceData;
import com.ruoyi.iot.domain.HistoryDeviceData;
import com.ruoyi.iot.domain.IotDeviceDetail;
import com.ruoyi.iot.domain.IotDeviceInfo;
import com.ruoyi.iot.domain.response.CommandResponse;
import com.ruoyi.iot.interfaces.IotDevice;
@ -145,4 +142,14 @@ public interface IotService {
CommandResponse setWifi(IotDevice device, String wifiName, String wifiPwd);
CommandResponse setWifi(String mac, String productId, String wifiName, String wifiPwd);
/**
* 设置倒计时提醒
*
* @param device 设备
* @param seconds 倒计时
*/
boolean setVoice(IotDevice device, long seconds);
boolean setVoice(String mac, String productId, long seconds);
}

View File

@ -31,8 +31,6 @@ import java.util.List;
import java.util.Objects;
import java.util.stream.Collectors;
import static org.bouncycastle.oer.its.Duration.seconds;
/**
* @author wjh
* 2024/3/20
@ -452,5 +450,31 @@ public class IotServiceImpl implements IotService {
return null;
}
@Override
public boolean setVoice(IotDevice device, long seconds) {
if (device == null) {
return false;
}
try {
boolean res = this.setVoice(device.iotMac1(), device.getProductId(), seconds);
ServiceUtil.assertion(!res, "设备MAC1设置语音播报时间失败");
return true;
} catch (Exception e) {
log.warn("设备MAC1设置语音播报时间失败: {}", e.getMessage());
return this.setVoice(device.iotMac2(), device.getProductId(), seconds);
}
}
@Override
public boolean setVoice(String mac, String productId, long seconds) {
if (StringUtils.isBlank(mac) || StringUtils.isBlank(productId)) {
return false;
}
CommandResponse res = this.sendCommand(mac, IotConstants.COMMAND_SET_VOICE + seconds + IotConstants.COMMAND_SEPARATOR, productId);
ServiceUtil.assertion(!res.isSuccess(), "设置语音播报失败:" + res.getMsg());
return res.isSuccess();
}
}

View File

@ -2,6 +2,7 @@ package com.ruoyi.ss.device.service;
import com.ruoyi.common.core.domain.ValidateResult;
import com.ruoyi.ss.device.domain.DeviceBO;
import com.ruoyi.ss.device.domain.vo.DeviceVO;
import java.util.Collection;
import java.util.List;
@ -50,4 +51,9 @@ public interface DeviceValidator {
* 判断MAC是否重复
*/
boolean isRepeatMac(Long deviceId, String mac);
/**
* 后校验
*/
void afterCheck(DeviceVO vo);
}

View File

@ -167,24 +167,26 @@ public class DeviceServiceImpl implements DeviceService
ServiceUtil.assertion(StringUtils.isBlank(data.getMac()), "参数错误MAC不允许为空");
String key = data.getDeviceNo();
ServiceUtil.assertion(!redisLock.lock(RedisLockKey.ADD_DEVICE, key), "当前录入人数过多,请稍后再试");
ServiceUtil.assertion(!redisLock.lock(RedisLockKey.ADD_DEVICE, key), "当前新增设备过于频繁,请稍后再试");
try {
ServiceUtil.assertion(deviceValidator.isRepeatMac(data.getDeviceId(), data.getMac()), "MAC-1重复:" + data.getMac());
ServiceUtil.assertion(deviceValidator.isRepeatMac(data.getDeviceId(), data.getMac2()), "MAC-2重复:" + data.getMac2());
ServiceUtil.assertion(deviceValidator.isRepeatSn(data.getDeviceId(), data.getDeviceNo()), "SN重复");
SmModelVO model = modelService.selectSmModelByModelId(data.getModelId());
ServiceUtil.assertion(model == null, "型号不存在");
ServiceUtil.assertion(StringUtils.isBlank(model.getProductId()), "型号产品ID为空");
ServiceUtil.assertion(StringUtils.isBlank(model.getProductId()), "型号产品ID未配置,请联系管理员处理");
data.setCreateTime(DateUtils.getNowDate());
data.setStatus(DeviceStatus.NORMAL.getStatus());
data.setOnlineStatus(DeviceOnlineStatus.OFFLINE.getStatus());
Integer result = transactionTemplate.execute(status -> {
// 创建设备
int insert = deviceMapper.insertSmDevice(data);
ServiceUtil.assertion(insert != 1, "新增设备失败");
// 后校验
DeviceVO vo = this.selectSmDeviceByDeviceId(data.getDeviceId());
deviceValidator.afterCheck(vo);
// 创建OneNet设备1
int code = iotService.create(data.getMac(), model.getProductId());
ServiceUtil.assertion(!IotHttpStatus.SUCCESS.equalCode(code) && !IotHttpStatus.DEVICE_EXIST.equalCode(code), "MAC-1注册失败");
@ -209,11 +211,7 @@ public class DeviceServiceImpl implements DeviceService
*/
@Override
@Transactional
public int updateSmDevice(DeviceBO data)
{
if (StringUtils.hasText(data.getDeviceNo())) {
ServiceUtil.assertion(deviceValidator.isRepeatSn(data.getDeviceId(), data.getDeviceNo()), "SN重复");
}
public int updateSmDevice(DeviceBO data) {
data.setUpdateTime(DateUtils.getNowDate());
@ -222,6 +220,10 @@ public class DeviceServiceImpl implements DeviceService
int update = deviceMapper.updateSmDevice(data);
ServiceUtil.assertion(update != 1, "更新设备失败");
// 后校验
DeviceVO vo = this.selectSmDeviceByDeviceId(data.getDeviceId());
deviceValidator.afterCheck(vo);
// 更新套餐关联
if (data.getSuitIds() != null) {
// 删除设备套餐关联

View File

@ -3,6 +3,8 @@ package com.ruoyi.ss.device.service.impl;
import com.ruoyi.common.core.domain.BaseValidator;
import com.ruoyi.common.core.domain.ValidateResult;
import com.ruoyi.common.utils.SecurityUtils;
import com.ruoyi.common.utils.ServiceUtil;
import com.ruoyi.common.utils.StringUtils;
import com.ruoyi.common.utils.collection.CollectionUtils;
import com.ruoyi.ss.device.domain.DeviceBO;
import com.ruoyi.ss.device.domain.DeviceQuery;
@ -11,7 +13,6 @@ import com.ruoyi.ss.device.service.DeviceService;
import com.ruoyi.ss.device.service.DeviceValidator;
import com.ruoyi.ss.store.service.StoreValidator;
import com.ruoyi.ss.suit.service.SuitValidator;
import org.apache.commons.lang.StringUtils;
import org.eclipse.jetty.util.StringUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@ -159,4 +160,21 @@ public class DeviceValidatorImpl extends BaseValidator implements DeviceValidato
}
return false;
}
@Override
public void afterCheck(DeviceVO vo) {
if (vo == null) {
return;
}
if (StringUtils.hasText(vo.getMac()) || StringUtils.hasText(vo.getMac2())) {
ServiceUtil.assertion(Objects.equals(vo.getMac(), vo.getMac2()), "MAC-1和MAC-2不允许相同");
}
if (vo.getUserId() != null || vo.getAgentId() != null) {
ServiceUtil.assertion(Objects.equals(vo.getUserId(), vo.getAgentId()), "代理商和商户不允许是同一个用户");
}
ServiceUtil.assertion(this.isRepeatMac(vo.getDeviceId(), vo.getMac()), "MAC-1重复:" + vo.getMac());
ServiceUtil.assertion(this.isRepeatMac(vo.getDeviceId(), vo.getMac2()), "MAC-2重复:" + vo.getMac2());
ServiceUtil.assertion(this.isRepeatSn(vo.getDeviceId(), vo.getDeviceNo()), "SN重复");
}
}

View File

@ -5,6 +5,9 @@ import com.ruoyi.common.core.redis.enums.RedisLockKey;
import com.ruoyi.common.exception.ServiceException;
import com.ruoyi.common.pay.syb.enums.SybTrxStatus;
import com.ruoyi.common.pay.syb.service.SybPayService;
import com.ruoyi.common.pay.tm.TmPayService;
import com.ruoyi.common.pay.tm.enums.TmPayStatus;
import com.ruoyi.common.pay.tm.vo.TmTradeInfo;
import com.ruoyi.common.pay.wx.service.WxPayService;
import com.ruoyi.common.pay.wx.util.WxPayUtil;
import com.ruoyi.common.utils.DateUtils;
@ -82,6 +85,9 @@ public class PayBillServiceImpl implements PayBillService
@Autowired
private SybPayService sybPayService;
@Autowired
private TmPayService tmPayService;
/**
* 查询支付订单
*
@ -239,6 +245,10 @@ public class PayBillServiceImpl implements PayBillService
}
}
}
// 太米微信
else if (TransactionBillPayType.TM_WX.getType().equals(bill.getChannelId())) {
tmPayService.closeOrder(bill.getPayNo());
}
}
}
@ -334,6 +344,8 @@ public class PayBillServiceImpl implements PayBillService
vo.setPayParams(wxPayService.prepayWithRequestPayment(bill));
} else if (TransactionBillPayType.TL_WX.getType().equals(bill.getChannelId())) {
vo.setPayParams(sybPayService.prepayWxApp(bill));
} else if (TransactionBillPayType.TM_WX.getType().equals(bill.getChannelId())) {
vo.setPayParams(tmPayService.pay(bill));
} else {
throw new ServiceException("暂不支持该支付方式");
}
@ -518,6 +530,14 @@ public class PayBillServiceImpl implements PayBillService
} else {
return PayResultVO.fail("暂未支付成功");
}
} else if (TransactionBillPayType.TM_WX.getType().equals(bill.getChannelId())) {
// 太米微信支付
TmTradeInfo result = tmPayService.orderQuery(bill.getPayNo());
if (result != null && TmPayStatus.isSuccess(result.getPayStatus())) {
return PayResultVO.success(LocalDateTime.now()); // TODO 支付时间
} else {
return PayResultVO.fail("暂未支付成功");
}
}
return PayResultVO.fail("暂不支持该支付方式");
}

View File

@ -3,6 +3,9 @@ package com.ruoyi.ss.refund.service.impl;
import com.ruoyi.common.exception.ServiceException;
import com.ruoyi.common.pay.syb.enums.SybTrxStatus;
import com.ruoyi.common.pay.syb.service.SybPayService;
import com.ruoyi.common.pay.tm.TmPayService;
import com.ruoyi.common.pay.tm.enums.TmRefundStatus;
import com.ruoyi.common.pay.tm.vo.RefundInfo;
import com.ruoyi.common.pay.wx.service.WxPayService;
import com.ruoyi.common.utils.DateUtils;
import com.ruoyi.common.utils.ServiceUtil;
@ -56,6 +59,9 @@ public class RefundServiceImpl implements RefundService
@Autowired
private ScheduledExecutorService scheduledExecutorService;
@Autowired
private TmPayService tmPayService;
/**
* 查询退款订单
@ -204,7 +210,17 @@ public class RefundServiceImpl implements RefundService
this.handleRefundSuccess(refund.getRefundNo());
}, 0, TimeUnit.SECONDS);
}
} else {
}
// 太米支付
else if (TransactionBillPayType.TM_WX.getType().equals(refund.getChannelId())) {
RefundInfo refundResult = tmPayService.refund(refundVO);
ServiceUtil.assertion(!TmRefundStatus.isSuccess(refundResult.getRefundStatus().getCode()), "发起退款失败:" + refundResult.getRefundMessage());
// 太米退款是同步通知直接处理退款成功
scheduledExecutorService.schedule(() -> {
handleRefundSuccess(refund.getRefundNo());
}, 0, TimeUnit.SECONDS);
}
else {
throw new ServiceException("当前支付方式不支持退款");
}

View File

@ -30,6 +30,7 @@ public class StoreBO extends Store {
bo.setContactName(getContactName());
bo.setContactMobile(getContactMobile());
bo.setShow(getShow());
bo.setUseOutTime(getUseOutTime());
return bo;
}
@ -52,6 +53,7 @@ public class StoreBO extends Store {
bo.setContactName(getContactName());
bo.setContactMobile(getContactMobile());
bo.setShow(getShow());
bo.setUseOutTime(getUseOutTime());
return bo;
}
@ -76,6 +78,7 @@ public class StoreBO extends Store {
bo.setContactName(getContactName());
bo.setContactMobile(getContactMobile());
bo.setShow(getShow());
bo.setUseOutTime(getUseOutTime());
return bo;
}
@ -102,6 +105,7 @@ public class StoreBO extends Store {
bo.setShow(getShow());
bo.setStatus(StoreStatus.NORMAL.getStatus());
bo.setEnabled(true);
bo.setUseOutTime(getUseOutTime());
return bo;
}
}

View File

@ -114,4 +114,14 @@ public class Suit extends BaseEntity
@Min(value = 0, message = "低功率关闭订单的功率值不允许小于0")
@JsonView(JsonViewProfile.App.class)
private BigDecimal lowPower;
@Excel(name = "是否开启结束前语音播报")
@ApiModelProperty("是否开启结束前语音播报")
@JsonView(JsonViewProfile.App.class)
private Boolean enabledVoice;
@Excel(name = "语音播报剩余时长", readConverterExp = "分=钟")
@ApiModelProperty("语音播报剩余时长(分钟)")
@JsonView(JsonViewProfile.App.class)
private BigDecimal voiceMinutes;
}

View File

@ -35,6 +35,8 @@ public class SuitBO extends Suit {
bo.setGearTime(getGearTime());
bo.setEnabledLowPowerClose(getEnabledLowPowerClose());
bo.setLowPower(getLowPower());
bo.setEnabledVoice(getEnabledVoice());
bo.setVoiceMinutes(getVoiceMinutes());
return bo;
}
@ -57,6 +59,8 @@ public class SuitBO extends Suit {
bo.setGearTime(getGearTime());
bo.setEnabledLowPowerClose(getEnabledLowPowerClose());
bo.setLowPower(getLowPower());
bo.setEnabledVoice(getEnabledVoice());
bo.setVoiceMinutes(getVoiceMinutes());
return bo;
}
}

View File

@ -30,6 +30,8 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
ss.gear_time,
ss.enabled_low_power_close,
ss.low_power,
ss.enabled_voice,
ss.voice_minutes,
su.user_name as user_name
from <include refid="searchTables"/>
</sql>
@ -48,9 +50,10 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="deleted == null"> and ss.deleted = false</if>
<if test="deleted != null"> and ss.deleted = #{deleted}</if>
<if test="userId != null"> and ss.user_id = #{userId}</if>
<if test="feeMode != null and feeMode != ''"> and fee_mode = #{feeMode}</if>
<if test="feeType != null and feeType != ''"> and fee_type = #{feeType}</if>
<if test="enabledLowPowerClose != null "> and enabled_low_power_close = #{enabledLowPowerClose}</if>
<if test="feeMode != null and feeMode != ''"> and ss.fee_mode = #{feeMode}</if>
<if test="feeType != null and feeType != ''"> and ss.fee_type = #{feeType}</if>
<if test="enabledLowPowerClose != null "> and ss.enabled_low_power_close = #{enabledLowPowerClose}</if>
<if test="enabledVoice != null "> and ss.enabled_voice = #{enabledVoice}</if>
<if test="deviceId != null">
and ss.suit_id in (
select distinct sds.suit_id from ss_device_suit sds where sds.device_id = #{deviceId}
@ -125,6 +128,8 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="gearTime != null">gear_time,</if>
<if test="enabledLowPowerClose != null">enabled_low_power_close,</if>
<if test="lowPower != null">low_power,</if>
<if test="enabledVoice != null">enabled_voice,</if>
<if test="voiceMinutes != null">voice_minutes,</if>
</trim>
<trim prefix="values (" suffix=")" suffixOverrides=",">
<if test="name != null and name != ''">#{name},</if>
@ -145,6 +150,8 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="gearTime != null">#{gearTime,typeHandler=com.ruoyi.system.mapper.typehandler.IntegerSplitListTypeHandler},</if>
<if test="enabledLowPowerClose != null">#{enabledLowPowerClose},</if>
<if test="lowPower != null">#{lowPower},</if>
<if test="enabledVoice != null">#{enabledVoice},</if>
<if test="voiceMinutes != null">#{voiceMinutes},</if>
</trim>
</insert>
@ -172,6 +179,8 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="gearTime != null">gear_time = #{gearTime,typeHandler=com.ruoyi.system.mapper.typehandler.IntegerSplitListTypeHandler},</if>
<if test="enabledLowPowerClose != null">enabled_low_power_close = #{enabledLowPowerClose},</if>
<if test="lowPower != null">low_power = #{lowPower},</if>
<if test="enabledVoice != null">enabled_voice = #{enabledVoice},</if>
<if test="voiceMinutes != null">voice_minutes = #{voiceMinutes},</if>
</sql>
<update id="logicDel">

View File

@ -290,6 +290,22 @@ public class TransactionBill extends BaseEntity implements Payable
@ApiModelProperty("代理商手机号")
private String agentMobile;
@Excel(name = "套餐是否开启语音播报")
@ApiModelProperty("套餐是否开启语音播报")
private Boolean suitEnabledVoid;
@Excel(name = "套餐语音播报阈值", readConverterExp = "分=钟")
@ApiModelProperty("套餐语音播报阈值")
private BigDecimal suitVoidMinute;
@Excel(name = "套餐语音播报设置结果", readConverterExp = "1=-成功2-失败")
@ApiModelProperty("套餐语音播报设置结果")
private String suitVoidResult;
@Excel(name = "套餐语音播报设置结果描述")
@ApiModelProperty("套餐语音播报设置结果描述")
private String suitVoidMsg;
/**
* 获取价格
*/

View File

@ -0,0 +1,21 @@
package com.ruoyi.ss.transactionBill.domain.enums;
import lombok.AllArgsConstructor;
import lombok.Getter;
/**
* 语音开启状态
* @author wjh
* 2024/9/30
*/
@Getter
@AllArgsConstructor
public enum RechargeVoiceStatus {
SUCCESS("1", "成功"),
FAIL("0", "失败");
private final String status;
private final String msg;
}

View File

@ -24,8 +24,8 @@ public enum TransactionBillPayType {
ALI(2L, "支付宝", AccountType.ALIPAY),
BANK(3L, "银行卡", AccountType.BANK_CARD),
BALANCE(4L, "余额支付", null),
TL_WX(5L, "通联微信支付", null)
;
TL_WX(5L, "通联微信支付", null),
TM_WX(6L, "太米微信支付", null);
private final Long type;
private final String name;
@ -44,7 +44,10 @@ public enum TransactionBillPayType {
return Arrays.stream(types).map(TransactionBillPayType::getType).collect(Collectors.toList());
}
/**
* 微信支付列表
*/
public static List<Long> wxList() {
return asList(WECHAT, TL_WX);
return asList(WECHAT, TL_WX, TM_WX);
}
}

View File

@ -72,6 +72,10 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
stb.close_result,
stb.agent_id,
stb.agent_mobile,
stb.suit_enabled_void,
stb.suit_void_minute,
stb.suit_void_result,
stb.suit_void_msg,
</sql>
<sql id="selectSmTransactionBillVo">
@ -192,6 +196,8 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="query.closeStatus != null and query.closeStatus != ''"> and stb.close_status = #{query.closeStatus}</if>
<if test="query.agentId != null "> and stb.agent_id = #{query.agentId}</if>
<if test="query.agentMobile != null and query.agentMobile != ''"> and stb.agent_mobile like concat('%', #{query.agentMobile}, '%')</if>
<if test="query.suitEnabledVoid != null "> and stb.suit_enabled_void = #{query.suitEnabledVoid}</if>
<if test="query.suitVoidResult != null and query.suitVoidResult != ''"> and stb.suit_void_result = #{query.suitVoidResult}</if>
<if test="query.isUsing != null">
<if test="query.isUsing">
and <include refid="isUsing"/>
@ -468,6 +474,10 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="closeResult != null">close_result,</if>
<if test="agentId != null">agent_id,</if>
<if test="agentMobile != null">agent_mobile,</if>
<if test="suitEnabledVoid != null">suit_enabled_void,</if>
<if test="suitVoidMinute != null">suit_void_minute,</if>
<if test="suitVoidResult != null">suit_void_result,</if>
<if test="suitVoidMsg != null">suit_void_msg,</if>
</trim>
<trim prefix="values (" suffix=")" suffixOverrides=",">
<if test="billNo != null">#{billNo},</if>
@ -528,6 +538,10 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="closeResult != null">#{closeResult},</if>
<if test="agentId != null">#{agentId},</if>
<if test="agentMobile != null">#{agentMobile},</if>
<if test="suitEnabledVoid != null">#{suitEnabledVoid},</if>
<if test="suitVoidMinute != null">#{suitVoidMinute},</if>
<if test="suitVoidResult != null">#{suitVoidResult},</if>
<if test="suitVoidMsg != null">#{suitVoidMsg},</if>
</trim>
</insert>
@ -606,6 +620,10 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="data.closeResult != null">stb.close_result = #{data.closeResult},</if>
<if test="data.agentId != null">stb.agent_id = #{data.agentId},</if>
<if test="data.agentMobile != null">stb.agent_mobile = #{data.agentMobile},</if>
<if test="data.suitEnabledVoid != null">stb.suit_enabled_void = #{data.suitEnabledVoid},</if>
<if test="data.suitVoidMinute != null">stb.suit_void_minute = #{data.suitVoidMinute},</if>
<if test="data.suitVoidResult != null">stb.suit_void_result = #{data.suitVoidResult},</if>
<if test="data.suitVoidMsg != null">stb.suit_void_msg = #{data.suitVoidMsg},</if>
</sql>
<update id="updateByQuery">

View File

@ -447,6 +447,8 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
order.setSuitName(suit.getName());
order.setSuitEnableLowPowerClose(suit.getEnabledLowPowerClose());
order.setSuitLowPower(suit.getLowPower());
order.setSuitEnabledVoid(suit.getEnabledVoice());
order.setSuitVoidMinute(suit.getVoiceMinutes());
// 设备信息
order.setDeviceNo(dto.getDeviceNo());
@ -954,14 +956,65 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
return updateCount;
});
// 充值设备尝试1次
// 操作成功
if (result != null && result == 1) {
// 充值设备尝试1次
this.tryRechargeDevice(bill.getBillId(), 1);
// 设备设置语音播报
scheduledExecutorService.schedule(() -> {
this.trySetVoice(bill, 3);
}, 0, TimeUnit.SECONDS);
}
return result == null ? 0 : result;
}
/**
* 尝试设置语音播报
*/
private void trySetVoice(TransactionBillVO bill, int tryCount) {
if (tryCount <= 0) {
return;
}
try {
transactionTemplate.execute(status -> {
// 设置为成功
int update = this.updateVoiceResult(bill.getBillId(), RechargeVoiceStatus.SUCCESS, "成功");
ServiceUtil.assertion(update != 1, "更新订单信息失败");
// 发送命令
boolean res = false;
if (bill.getSuitEnabledVoid() == null || !bill.getSuitEnabledVoid() || bill.getSuitVoidMinute() == null) {
res = iotService.setVoice(bill, 0L);
} else {
res = iotService.setVoice(bill, bill.getSuitVoidMinute().multiply(BigDecimal.valueOf(60)).longValue());
}
ServiceUtil.assertion(!res, "设备设置语音失败");
return res;
});
} catch (Exception e) {
log.info("订单设置设备语音失败:{}, 剩余次数:{}", bill.getBillNo(), tryCount - 1);
if (tryCount <= 1) {
// 设置为失败
int update = this.updateVoiceResult(bill.getBillId(), RechargeVoiceStatus.FAIL, e.getMessage());
ServiceUtil.assertion(update != 1, "更新订单信息失败");
} else {
this.trySetVoice(bill, tryCount - 1);
}
}
}
private int updateVoiceResult(Long billId, RechargeVoiceStatus status, String msg) {
TransactionBill data = new TransactionBill();
data.setBillId(billId);
data.setSuitVoidResult(status.getStatus());
data.setSuitVoidMsg(msg);
return transactionBillMapper.updateSmTransactionBill(data);
}
/**
* 异步尝试充值设备
* @param billId
@ -971,14 +1024,12 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
if (tryCount <= 0) {
return;
}
// scheduledExecutorService.schedule(()-> {
try {
boolean result = rechargeDevice(billId);
ServiceUtil.assertion(!result, String.format("尝试充值设备失败:billId=%s:剩余次数:%s,", billId, tryCount - 1));
} catch (Exception e) {
this.tryRechargeDevice(billId, tryCount - 1);
}
// }, 0, TimeUnit.SECONDS);
}
@Override

View File

@ -71,7 +71,7 @@ public class TransactionBillValidatorImpl extends BaseValidator implements Trans
if (store != null) {
LocalTime now = LocalTime.now();
if (!storeValidator.isBusinessTime(store, now)) {
return error(String.format("当前店铺不在营业时间内,无法下单。营业时间:%s - %s", store.getBusinessTimeStart(), store.getBusinessTimeEnd()));
return error(String.format("请在营业时间内使用该设备!营业时间:%s - %s", store.getBusinessTimeStart(), store.getBusinessTimeEnd()));
}
}

View File

@ -1,6 +1,7 @@
package com.ruoyi.web.controller.app;
import com.alibaba.fastjson2.JSON;
import com.alibaba.fastjson2.JSONObject;
import com.ruoyi.common.annotation.Anonymous;
import com.ruoyi.common.core.controller.BaseController;
import com.ruoyi.common.core.domain.AjaxResult;
@ -9,6 +10,8 @@ import com.ruoyi.common.pay.syb.enums.SybTrxCode;
import com.ruoyi.common.pay.syb.enums.SybTrxStatus;
import com.ruoyi.common.pay.syb.service.SybPayService;
import com.ruoyi.common.pay.syb.util.SybUtil;
import com.ruoyi.common.pay.tm.TmPayService;
import com.ruoyi.common.pay.tm.enums.TmPayStatus;
import com.ruoyi.common.pay.wx.domain.enums.WxNotifyEventType;
import com.ruoyi.common.pay.wx.domain.enums.WxTransferBatchStatus;
import com.ruoyi.common.pay.wx.util.WxPayUtil;
@ -38,6 +41,8 @@ import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.*;
import javax.servlet.http.HttpServletRequest;
import java.time.LocalDateTime;
import java.util.Map;
import java.util.TreeMap;
/**
@ -65,6 +70,9 @@ public class AppPayController extends BaseController {
@Autowired
private TransactionBillConverter transactionBillConverter;
@Autowired
private TmPayService tmPayService;
@ApiOperation("微信支付充值订单")
@GetMapping("/wx/{billNo}")
public AjaxResult wxPay(@PathVariable @ApiParam("订单编号") String billNo) {
@ -192,4 +200,35 @@ public class AppPayController extends BaseController {
}
}
/**
* 太米微信支付回调
*/
@ApiOperation(value = "太米微信支付回调")
@PostMapping("/notify/tm")
public String tmwx(HttpServletRequest request) {
try {
String body = HttpUtils.getBody(request);
log.info("【太米微信支付回调】接收对象 : " + body);
// 先把body转成map
Map<String, Object> params = JSON.parseObject(body, Map.class);
// 验证签名
boolean sign = tmPayService.validSign(params);
if (sign) {
JSONObject tradeInfo = (JSONObject)params.get("tradeInfo");
String payType = tradeInfo.getString("payType"); // 交易类型
String outTradeId = tradeInfo.getString("outTradeId"); // 商户自定义订单号
String payStatus = tradeInfo.getString("payStatus"); // 交易结果
if(TmPayStatus.isSuccess(payStatus) && payType.equals("wx_pay")) {
payBillService.handleSuccess(outTradeId, LocalDateTime.now()); // TODO 支付时间待定
}
}else {
throw new ServiceException("签名验证失败");
}
return "{\"result\":\"SUCCESS\"}";
} catch (Exception e) {
throw new ServiceException(e.getMessage());
}
}
}

View File

@ -282,7 +282,6 @@ public class AppTransactionBillController extends BaseController
@ApiOperation("支付押金")
@PutMapping("/payDeposit")
public AjaxResult payDeposit(@RequestBody @Validated PayDepositDTO dto) {
dto.setChannelId(1L);
return success(transactionBillService.payDeposit(transactionBillConverter.toRechargePayDepositBO(dto)));
}
@ -355,7 +354,6 @@ public class AppTransactionBillController extends BaseController
@ApiOperation("支付订单")
@PutMapping("/pay")
public AjaxResult pay(@RequestBody BillPayDTO dto) {
dto.setChannelId(1L);
TransactionBillVO bill = transactionBillService.selectSmTransactionBillByBillNo(dto.getBillNo());
if (!transactionBillValidator.isUser(bill, getUserId())) {
return error("这不是您的订单,无法支付");

View File

@ -110,6 +110,15 @@ syb:
# 支付通知地址
notifyUrl: http://124.221.246.124:2290/app/pay/notify/tl
# 太米支付
tm:
developerId: "100232"
shopId: "0034947"
signKey: b4ixpiogfj5vu3tbkv23gj0dvo2j2ksz
httpUrl: https://v5.taimi100.com
sn: ""
notifyUrl: http://124.221.246.124:2290/app/pay/notify/tm
# 活体检测跳转地址
liveness:
returnUrl: http://192.168.2.81:3001/liveness