双模设备、用户密码、蓝牙设备充值、支付回调优化

This commit is contained in:
墨大叔 2024-09-25 18:30:23 +08:00
parent 49469ba6f1
commit e2b9f18a21
36 changed files with 612 additions and 215 deletions

View File

@ -15,8 +15,8 @@ import java.util.Objects;
@AllArgsConstructor
public enum UserType {
TENANT("00", ""),
LANDLORD("01", "商户");
NORMAL("00", "普通用"),
MCH("01", "商户");
private final String type;
private final String name;

View File

@ -691,4 +691,15 @@ public class StringUtils extends org.apache.commons.lang3.StringUtils
}
return new String(chars);
}
/**
* 替换最后一个
* @param str
* @param pattern
* @param replace
* @return
*/
public static String replaceLast(String str, String pattern, String replace) {
return reverse(reverse(str).replaceFirst(reverse(pattern), reverse(replace)));
}
}

View File

@ -72,7 +72,7 @@ public class HttpUtils
try
{
String urlNameString = StringUtils.isNotBlank(param) ? url + "?" + param : url;
log.info("sendGet - {}", urlNameString);
// log.info("sendGet - {}", urlNameString);
URL realUrl = new URL(urlNameString);
URLConnection connection = realUrl.openConnection();
connection.setRequestProperty("accept", "*/*");
@ -382,7 +382,7 @@ public class HttpUtils
try
{
String urlNameString = StringUtils.isNotBlank(param) ? url + "?" + param : url;
log.info("sendGet - {}", urlNameString);
// log.info("sendGet - {}", urlNameString);
URL realUrl = new URL(urlNameString);
URLConnection connection = realUrl.openConnection();
connection.setRequestProperty("accept", "*/*");

View File

@ -2,10 +2,10 @@ package com.ruoyi.iot.domain;
import com.ruoyi.common.constant.IotConstants;
import com.ruoyi.common.utils.NumberUtils;
import com.ruoyi.common.utils.collection.CollectionUtils;
import com.ruoyi.iot.constants.ReceiveConstants;
import com.ruoyi.ss.device.domain.enums.DeviceOutageWay;
import lombok.Data;
import org.springframework.util.CollectionUtils;
import java.math.BigDecimal;
import java.math.RoundingMode;
@ -27,12 +27,13 @@ public class CurrentDeviceData {
* @return
*/
public IotDeviceInfo parseDeviceInfo() {
if (CollectionUtils.isEmptyElement(datastreams)) {
return null;
}
IotDeviceInfo device = IotDeviceInfo.newDefaultInstance();
device.setMac(title);
device.setId(id);
if (CollectionUtils.isEmpty(datastreams)) {
return device;
}
for (CurrentDatastream stream : datastreams) {
String value = stream.getValue().toString();
@ -51,7 +52,7 @@ public class CurrentDeviceData {
device.setW(NumberUtils.nonNullDecimal(value).divide(new BigDecimal(1000), 2, RoundingMode.HALF_UP));
break;
case ReceiveConstants.DS_S:
device.setS(value);
device.setS(Integer.valueOf(value).toString());
break;
case ReceiveConstants.DS_M:
device.setM(NumberUtils.nonNullDecimal(value).divide(new BigDecimal(1000), 2, RoundingMode.HALF_UP));

View File

@ -29,7 +29,7 @@ public class IotDeviceInfo {
private BigDecimal w; // 总用电量
private String s; // 开关状态0不通电1闭合通电
private BigDecimal m; // 剩余电量
private String set; // 欠费断电方式
private String set; // 开关设置状态
private BigDecimal time; // 剩余时间
private String model; // 型号
private String wifi; // WIFI

View File

@ -0,0 +1,18 @@
package com.ruoyi.iot.interfaces;
/**
* @author wjh
* 2024/9/25
*/
public interface IotDevice {
// 获取MAC-1
String getMac1();
// 获取MAC-2
String getMac2();
// 获取OneNet产品ID
String getProductId();
}

View File

@ -97,7 +97,7 @@ public class IotReceiveServiceImpl implements IotReceiveService{
}
if (seconds > 0) {
iotService.setTime(device.getMac(), seconds, device.getModelProductId());
iotService.setTime(device, seconds);
}
if (ele.compareTo(BigDecimal.ZERO) > 0) {
iotService.setEle(device.getMac(), ele, device.getModelProductId());

View File

@ -6,7 +6,7 @@ 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.ss.device.domain.Device;
import com.ruoyi.iot.interfaces.IotDevice;
import com.ruoyi.ss.device.domain.enums.DeviceOnlineStatus;
import java.math.BigDecimal;
@ -17,28 +17,6 @@ import java.util.List;
* 2024/3/20
*/
public interface IotService {
/**
* 向设备发送命令
*
* @param deviceName OneNet设备名称即设备表的MAC号
* @param command 命令字符串
* @param productId
* @return
*/
default CommandResponse sendCommand(String deviceName, String command, String productId) {
return sendCommand(deviceName, command, null, productId);
}
/**
* 向设备发送命令
*
* @param deviceName OneNet设备名称即设备表的MAC号
* @param command 命令字符串
* @param timeout
* @param productId
* @return
*/
CommandResponse sendCommand(String deviceName, String command, Integer timeout, String productId);
/**
* 获取设备在线状态
@ -48,6 +26,11 @@ public interface IotService {
*/
DeviceOnlineStatus getOnlineStatus(String deviceName, String productId);
/**
* 获取设备在线状态
*/
DeviceOnlineStatus getOnlineStatus(IotDevice device);
/**
* 通电
*
@ -57,6 +40,10 @@ public interface IotService {
*/
boolean open(String deviceName, String productId);
/**
* 通电
*/
boolean open(IotDevice device);
/**
* 断电
@ -68,36 +55,9 @@ public interface IotService {
boolean close(String deviceName, String productId);
/**
* 获取历史设备数据点信息
*
* @param deviceName OneNet设备名称即设备表的MAC号
* @param productId
* 断电
*/
HistoryDeviceData getHistoryDataPoint(String deviceName, String productId);
/**
* 批量获取当前设备数据点信息
*
* @param deviceNames 设备名称列表
* @param productId
*/
List<CurrentDeviceData> getCurrentDataPoint(List<String> deviceNames, String productId);
/**
* 获取当前设备数据点信息
*
* @param deviceName 设备名称
* @param productId
*/
CurrentDeviceData getCurrentDataPoint(String deviceName, String productId);
/**
* 获取设备详情
*
* @param deviceName 设备名称
* @param productId
*/
IotDeviceDetail getDeviceDetail(String deviceName, String productId);
boolean close(IotDevice device);
/**
* 设置剩余时长
@ -109,12 +69,9 @@ public interface IotService {
CommandResponse setTime(String deviceName, long seconds, String productId);
/**
* 更新设备信息
*
* @param device 设备信息
* @param productId
* 设置剩余时长
*/
boolean updateDevice(Device device, String productId);
CommandResponse setTime(IotDevice device, long seconds);
/**
* 获取设备信息并转化为IotDeviceInfo
@ -124,6 +81,11 @@ public interface IotService {
*/
IotDeviceInfo getDeviceInfo(String deviceName, String productId);
/**
* 获取设备信息并转为IotDeviceInfo
*/
IotDeviceInfo getDeviceInfo(IotDevice device);
/**
* 获取设备信息列表并转换为IotDeviceInfo
*
@ -142,8 +104,18 @@ public interface IotService {
*/
CommandResponse addEle(String deviceName, BigDecimal ele, String productId);
/**
* 设备添加电量
*/
CommandResponse addEle(IotDevice device, BigDecimal ele);
/**
* 直接设置设备电量
*/
CommandResponse setEle(String deviceName, BigDecimal ele, String productId);
/**
* 直接设置设备电量
*/
CommandResponse setEle(IotDevice device, BigDecimal ele);
}

View File

@ -15,14 +15,11 @@ import com.ruoyi.iot.domain.response.CurrentDataPointResponse;
import com.ruoyi.iot.domain.response.DetailResponse;
import com.ruoyi.iot.domain.response.HistoryDataPointResponse;
import com.ruoyi.iot.enums.IotHttpStatus;
import com.ruoyi.iot.util.CommandBuilder;
import com.ruoyi.ss.device.domain.Device;
import com.ruoyi.iot.interfaces.IotDevice;
import com.ruoyi.ss.device.domain.enums.DeviceOnlineStatus;
import com.ruoyi.ss.device.domain.enums.DeviceOutageWay;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.util.CollectionUtils;
import java.math.BigDecimal;
@ -76,9 +73,29 @@ public class IotServiceImpl implements IotService {
return IotDeviceDetail.Status.toDeviceOnlineStatus(detail.getStatus());
}
@Override
public DeviceOnlineStatus getOnlineStatus(IotDevice device) {
DeviceOnlineStatus status = DeviceOnlineStatus.OFFLINE;
if (device == null || StringUtils.isBlank(device.getProductId())) {
return status;
}
// 优先使用mac1判断
if (StringUtils.hasText(device.getMac1())) {
status = this.getOnlineStatus(device.getMac1(), device.getProductId());
}
// 若还是离线则判断mac2是否在线
if (status == DeviceOnlineStatus.OFFLINE && StringUtils.hasText(device.getMac2())) {
status = this.getOnlineStatus(device.getMac2(), device.getProductId());
}
return status;
}
// 通电
@Override
public boolean open(String deviceName, String productId) {
if (StringUtils.hasBlank(deviceName, productId)) {
return false;
}
CommandResponse response = sendCommand(deviceName, IotConstants.COMMAND_OPEN, productId);
IotHttpStatus status = IotHttpStatus.convertByCode(response.getCode());
if (!IotHttpStatus.SUCCESS.equals(status)) {
@ -87,6 +104,27 @@ public class IotServiceImpl implements IotService {
return true;
}
@Override
public boolean open(IotDevice device) {
boolean result = false;
if (device == null || StringUtils.isBlank(device.getProductId())) {
return result;
}
// 尝试用mac1通电
try {
result = this.open(device.getMac1(), device.getProductId());
if (!result) {
throw new ServiceException("mac1通电失败");
}
} catch (Exception e) {
log.info("mac1通电失败尝试用mac2通电");
result = this.open(device.getMac2(), device.getProductId());
}
return result;
}
// 断电
@Override
public boolean close(String deviceName, String productId) {
@ -98,9 +136,29 @@ public class IotServiceImpl implements IotService {
return true;
}
// 获取历史设备数据点信息
@Override
public HistoryDeviceData getHistoryDataPoint(String deviceName, String productId) {
public boolean close(IotDevice device) {
boolean result = false;
if (device == null || StringUtils.isBlank(device.getProductId())) {
return result;
}
// 尝试用mac1断电
try {
result = this.close(device.getMac1(), device.getProductId());
if (!result) {
throw new ServiceException("mac1断电失败");
}
} catch (Exception e) {
log.info("mac1断电失败尝试用mac2断电");
result = this.close(device.getMac2(), device.getProductId());
}
return result;
}
// 获取历史设备数据点信息
private HistoryDeviceData getHistoryDataPoint(String deviceName, String productId) {
String param = "device_name=" + deviceName + "&product_id=" + productId;
String sendUrl = iotHost + IotConstants.ADDS_HISTORY_DATAPOINTS + "?"+param;
@ -122,7 +180,6 @@ public class IotServiceImpl implements IotService {
}
// 获取当前设备数据点信息
@Override
public List<CurrentDeviceData> getCurrentDataPoint(List<String> deviceNames, String productId) {
String param = "device_name=" + String.join(",", deviceNames) + "&product_id=" + productId;
String sendUrl = iotHost+ IotConstants.ADDS_CURRENT_DATAPOINTS + "?" + param;
@ -144,8 +201,7 @@ public class IotServiceImpl implements IotService {
}
// 获取当前设备数据点信息
@Override
public CurrentDeviceData getCurrentDataPoint(String deviceName, String productId) {
private CurrentDeviceData getCurrentDataPoint(String deviceName, String productId) {
List<CurrentDeviceData> list = this.getCurrentDataPoint(Collections.singletonList(deviceName), productId);
if (CollectionUtils.isEmpty(list)) {
return null;
@ -153,8 +209,7 @@ public class IotServiceImpl implements IotService {
return list.stream().filter(item -> Objects.equals(item.getTitle(), deviceName)).findFirst().orElse(null);
}
@Override
public IotDeviceDetail getDeviceDetail(String deviceName, String productId) {
private IotDeviceDetail getDeviceDetail(String deviceName, String productId) {
String sendUrl = iotHost + IotConstants.ADDS_DEVICE_DETAIL;
String param = "device_name=" + deviceName + "&product_id=" + productId;
@ -191,6 +246,23 @@ public class IotServiceImpl implements IotService {
return sendCommand(deviceName, IotConstants.COMMAND_RECHARGE + seconds + IotConstants.COMMAND_SEPARATOR, 5, productId);
}
@Override
public CommandResponse setTime(IotDevice device, long seconds) {
if (device == null || StringUtils.isBlank(device.getProductId())) {
throw new ServiceException("设置时长错误:参数不允许为空");
}
CommandResponse res = null;
if (StringUtils.hasText(device.getMac1())) {
res = this.setTime(device.getMac1(), seconds, device.getProductId());
}
if ((res == null || !res.isSuccess()) && StringUtils.hasText(device.getMac2())) {
res = this.setTime(device.getMac2(), seconds, device.getProductId());
}
return res;
}
@Override
public CommandResponse addEle(String deviceName, BigDecimal ele, String productId) {
@ -200,32 +272,46 @@ public class IotServiceImpl implements IotService {
return sendCommand(deviceName, IotConstants.COMMAND_ADD_ELE + ele.multiply(BigDecimal.valueOf(1000)) + IotConstants.COMMAND_SEPARATOR, 5, productId);
}
@Override
public CommandResponse addEle(IotDevice device, BigDecimal ele) {
if (device == null || StringUtils.isBlank(device.getProductId())) {
throw new ServiceException("充值电量错误:参数不允许为空");
}
CommandResponse res = null;
if (StringUtils.hasText(device.getMac1())) {
res = this.addEle(device.getMac1(), ele, device.getProductId());
}
if ((res == null || !res.isSuccess()) && StringUtils.hasText(device.getMac2())) {
res = this.addEle(device.getMac2(), ele, device.getProductId());
}
return res;
}
@Override
public CommandResponse setEle(String deviceName, BigDecimal ele, String productId) {
if (ele == null) {
throw new ServiceException("设置电量错误:不允许为空");
throw new ServiceException("设置电量错误:电量不允许为空");
}
return sendCommand(deviceName, IotConstants.COMMAND_SET_ELE + ele.multiply(BigDecimal.valueOf(1000)) + IotConstants.COMMAND_SEPARATOR, 5, productId);
}
/**
* 更新设备信息
*
* @param device 设备信息
* @param productId
*/
@Override
@Transactional
public boolean updateDevice(Device device, String productId) {
DeviceOutageWay deviceOutageWay = DeviceOutageWay.parse(device.getOutageWay());
public CommandResponse setEle(IotDevice device, BigDecimal ele) {
if (device == null || StringUtils.isBlank(device.getProductId())) {
throw new ServiceException("设置电量错误:参数不允许为空");
}
String command = CommandBuilder.builder()
// 断电方式若为空则立即断电
.setIfNull(IotConstants.COMMAND_OUTAGE_WAY, deviceOutageWay.getValue() , DeviceOutageWay.IMMEDIATE.getValue())
.build();
CommandResponse response = sendCommand(device.getMac(), command, productId);
ServiceUtil.assertion(!Objects.equals(IotHttpStatus.SUCCESS.getCode(), response.getCode()), "修改设备设置发生异常:" + response.getMsg());
return true;
CommandResponse res = null;
if (StringUtils.hasText(device.getMac1())) {
res = this.setEle(device.getMac1(), ele, device.getProductId());
}
if ((res == null || !res.isSuccess()) && StringUtils.hasText(device.getMac2())) {
res = this.setEle(device.getMac2(), ele, device.getProductId());
}
return res;
}
@Override
@ -240,6 +326,23 @@ public class IotServiceImpl implements IotService {
return currentDataPoint.parseDeviceInfo();
}
@Override
public IotDeviceInfo getDeviceInfo(IotDevice device) {
IotDeviceInfo info = null;
if (device == null || StringUtils.isBlank(device.getProductId())) {
return info;
}
if (StringUtils.hasText(device.getMac1())) {
info = getDeviceInfo(device.getMac1(), device.getProductId());
}
if (info == null && StringUtils.hasText(device.getMac2())) {
info = getDeviceInfo(device.getMac2(), device.getProductId());
}
return info;
}
@Override
public List<IotDeviceInfo> getDeviceInfo(List<String> deviceNames, String productId) {
if (CollectionUtils.isEmpty(deviceNames)) {
@ -249,7 +352,10 @@ public class IotServiceImpl implements IotService {
if (CollectionUtils.isEmpty(dataList)) {
return Collections.emptyList();
}
return dataList.stream().map(CurrentDeviceData::parseDeviceInfo).collect(Collectors.toList());
return dataList.stream()
.map(CurrentDeviceData::parseDeviceInfo)
.filter(Objects::nonNull)
.collect(Collectors.toList());
}
@Override
@ -267,10 +373,12 @@ public class IotServiceImpl implements IotService {
return paramsObj.getInteger("code");
}
private CommandResponse sendCommand(String deviceName, String command, String productId) {
return sendCommand(deviceName, command, null, productId);
}
// 发送MQTT命令
@Override
public CommandResponse sendCommand(String deviceName, String command, Integer timeout, String productId) {
private CommandResponse sendCommand(String deviceName, String command, Integer timeout, String productId) {
if (timeout == null) {
timeout = this.timeout;
}

View File

@ -64,6 +64,11 @@ public class Device extends BaseEntity
@JsonView(JsonViewProfile.App.class)
private String mac;
@Excel(name = "设备Mac号2")
@ApiModelProperty("设备Mac号2")
@JsonView(JsonViewProfile.App.class)
private String mac2;
/** 激活时间 */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
@ApiModelProperty("激活时间")
@ -113,14 +118,8 @@ public class Device extends BaseEntity
@ApiModelProperty("用户昵称")
private String nickName;
@Excel(name = "欠费断电")
@ApiModelProperty("断电方式0欠费不断电1欠费立即断电")
// @EnumValid(
// clazz = DeviceOutageWay.class,
// message = "非法的超时断电方式",
// method = "getValue",
// groups = {ValidGroup.Create.class, ValidGroup.Update.class, ValidGroup.FrontUpdate.class}
// )
@Excel(name = "开关设置方式")
@ApiModelProperty("开关设置方式")
private String outageWay;
@Excel(name = "wifi名称")
@ -205,4 +204,9 @@ public class Device extends BaseEntity
@ApiModelProperty("限制充值原因")
@Size(max = 200, message = "限制充值原因长度不能超过200个字符")
private String limitRechargeReason;
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
@Excel(name = "最后在线时间", width = 30, dateFormat = "yyyy-MM-dd HH:mm:ss")
@ApiModelProperty("最后在线时间")
private LocalDateTime lastOnlineTime;
}

View File

@ -73,4 +73,10 @@ public class DeviceQuery extends Device {
@ApiModelProperty("MAC列表")
private List<String> macList;
@ApiModelProperty("任意MAC")
private String anyMac;
@ApiModelProperty("模糊查询任意MAC")
private String likeAnyMac;
}

View File

@ -10,6 +10,7 @@ import lombok.Data;
public class DeviceMacSnVO {
private String mac;
private String mac2;
private String sn;
}

View File

@ -3,6 +3,7 @@ package com.ruoyi.ss.device.domain.vo;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.annotation.JsonView;
import com.ruoyi.common.core.domain.JsonViewProfile;
import com.ruoyi.iot.interfaces.IotDevice;
import com.ruoyi.ss.device.domain.DeviceView;
import com.ruoyi.ss.device.domain.Device;
import com.ruoyi.ss.suit.domain.SuitVO;
@ -18,7 +19,7 @@ import java.util.List;
* 2024/3/15
*/
@Data
public class DeviceVO extends Device {
public class DeviceVO extends Device implements IotDevice {
@ApiModelProperty("是否默认")
private Boolean isDefault; // 是否默认
@ -74,4 +75,13 @@ public class DeviceVO extends Device {
@ApiModelProperty("型号产品ID")
private String modelProductId;
@Override
public String getMac1() {
return getMac();
}
@Override
public String getProductId() {
return getModelProductId();
}
}

View File

@ -98,7 +98,7 @@ public interface DeviceMapper
* @param deviceId 设备id
* @param storeId 店铺id
*/
int bindStore(@Param("deviceId") Long deviceId, @Param("storeId") Long storeId, @Param("userId") Long userId);
int bind(@Param("deviceId") Long deviceId, @Param("storeId") Long storeId, @Param("userId") Long userId);
/**
* 解绑商户

View File

@ -15,7 +15,18 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="serviceType != null and serviceType != ''"> and sd.service_type = #{serviceType}</if>
<if test="deviceName != null and deviceName != ''"> and sd.device_name like concat('%', #{deviceName}, '%')</if>
<if test="model != null and model != ''"> and sm.model_name like concat('%', #{model}, '%')</if>
<if test="mac != null and mac != ''"> and sd.mac like concat('%', #{mac}, '%')</if>
<if test="mac != null and mac != ''"> and sd.mac = #{mac}</if>
<if test="mac2 != null and mac2 != ''"> and sd.mac2 = #{mac2}</if>
<if test="anyMac != null and anyMac != ''">
and (
sd.mac = #{anyMac} or sd.mac2 = #{anyMac}
)
</if>
<if test="likeAnyMac != null and likeAnyMac != ''">
and (
sd.mac like concat('%', #{likeAnyMac}, '%') or sd.mac2 like concat('%', #{likeAnyMac}, '%')
)
</if>
<if test="onlineStatus != null and onlineStatus != ''"> and sd.online_status = #{onlineStatus}</if>
<if test="status != null and status != ''"> and sd.status = #{status}</if>
<if test="userName != null and userName != ''"> and su.user_name like concat('%', #{userName}, '%')</if>
@ -68,7 +79,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
</foreach>
</if>
<if test="modelIds != null and modelIds.size() > 0">
and sd.moidel_id in
and sd.model_id in
<foreach collection="modelIds" open="(" close=")" separator="," item="item">
#{item}
</foreach>
@ -91,6 +102,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
sd.device_name,
sd.model_id,
sd.mac,
sd.mac2,
sd.activation_time,
sd.total_electri_quantity,
sd.online_status,
@ -127,6 +139,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
sd.surplus_ele,
sd.limit_recharge_time,
sd.limit_recharge_reason,
sd.last_online_time,
sm.model_name as model,
sm.picture as picture,
sm.tags as model_tags,
@ -236,6 +249,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<select id="selectMacSnList" resultType="DeviceMacSnVO">
select
sd.mac,
sd.mac2,
sd.device_no as sn
from sm_device sd
<where>
@ -261,6 +275,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="deviceName != null">device_name,</if>
<if test="modelId != null">model_id,</if>
<if test="mac != null">mac,</if>
<if test="mac2 != null">mac2,</if>
<if test="activationTime != null">activation_time,</if>
<if test="totalElectriQuantity != null">total_electri_quantity,</if>
<if test="onlineStatus != null">online_status,</if>
@ -297,12 +312,14 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="surplusEle != null">surplus_ele,</if>
<if test="limitRechargeTime != null">limit_recharge_time,</if>
<if test="limitRechargeReason != null">limit_recharge_reason,</if>
<if test="lastOnlineTime != null">last_online_time,</if>
</trim>
<trim prefix="values (" suffix=")" suffixOverrides=",">
<if test="storeId != null">#{storeId},</if>
<if test="deviceName != null">#{deviceName},</if>
<if test="modelId != null">#{modelId},</if>
<if test="mac != null">#{mac},</if>
<if test="mac2 != null">#{mac2},</if>
<if test="activationTime != null">#{activationTime},</if>
<if test="totalElectriQuantity != null">#{totalElectriQuantity},</if>
<if test="onlineStatus != null">#{onlineStatus},</if>
@ -339,6 +356,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="surplusEle != null">#{surplusEle},</if>
<if test="limitRechargeTime != null">#{limitRechargeTime},</if>
<if test="limitRechargeReason != null">#{limitRechargeReason},</if>
<if test="lastOnlineTime != null">#{lastOnlineTime},</if>
</trim>
</insert>
@ -365,6 +383,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="deviceName != null">device_name = #{deviceName},</if>
<if test="modelId != null">model_id = #{modelId},</if>
<if test="mac != null">mac = #{mac},</if>
<if test="mac2 != null">mac2 = #{mac2},</if>
<if test="deviceNo != null">device_no = #{deviceNo},</if>
<if test="activationTime != null">activation_time = #{activationTime},</if>
<if test="totalElectriQuantity != null">total_electri_quantity = #{totalElectriQuantity},</if>
@ -402,6 +421,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="surplusEle != null">surplus_ele = #{surplusEle},</if>
<if test="limitRechargeTime != null">limit_recharge_time = #{limitRechargeTime},</if>
<if test="limitRechargeReason != null">limit_recharge_reason = #{limitRechargeReason},</if>
<if test="lastOnlineTime != null">last_online_time = #{lastOnlineTime},</if>
</trim>
where device_id = #{deviceId}
</update>
@ -415,7 +435,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
</foreach>
</update>
<update id="bindStore">
<update id="bind">
update sm_device
set store_id = #{storeId},
user_id = #{userId},
@ -483,6 +503,16 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
</otherwise>
</choose>
</foreach>
<foreach open="last_online_time = CASE device_id" collection="list" item="item" close="END,">
<choose>
<when test="item.lastOnlineTime != null">
WHEN #{item.deviceId} THEN #{item.lastOnlineTime}
</when>
<otherwise>
WHEN #{item.deviceId} THEN `last_online_time`
</otherwise>
</choose>
</foreach>
</trim>
where device_id in
<foreach item="item" collection="list" open="(" separator="," close=")">

View File

@ -143,7 +143,17 @@ public interface DeviceService
*
* @param deviceId 设备id
*/
int resetTimeWithBill(Long deviceId);
default int resetTimeWithBill(Long deviceId) {
return resetTimeWithBill(deviceId, true);
}
/**
* 归零电表并关闭订单
*
* @param deviceId 设备id
* @param requiredIot 是否操作设备
*/
int resetTimeWithBill(Long deviceId, boolean requiredIot);
/**
* 设备是否已经被绑定
@ -226,7 +236,7 @@ public interface DeviceService
*/
void deviceHeartBeat();
boolean addTimeByUser(Long deviceId, long seconds, boolean b, String reason);
boolean addTimeByUser(Long deviceId, long seconds, boolean withIot, String reason);
/**
* 设备注册
@ -307,6 +317,17 @@ public interface DeviceService
/**
* 电量归零并关闭订单
*/
int resetEleWithBill(Long deviceId);
default int resetEleWithBill(Long deviceId) {
return resetEleWithBill(deviceId, true);
}
/**
* 电量归零并关闭订单
*/
int resetEleWithBill(Long deviceId, boolean requiredIot);
/**
* 添加电量
*/
boolean addEle(Long deviceId, BigDecimal amount, boolean withIot, String reason);
}

View File

@ -5,6 +5,7 @@ import com.ruoyi.common.core.domain.model.LoginUser;
import com.ruoyi.common.core.redis.RedisLock;
import com.ruoyi.common.core.redis.enums.RedisLockKey;
import com.ruoyi.common.enums.LoginType;
import com.ruoyi.common.enums.UserType;
import com.ruoyi.common.exception.ServiceException;
import com.ruoyi.common.utils.*;
import com.ruoyi.common.utils.collection.CollectionUtils;
@ -125,6 +126,9 @@ public class DeviceServiceImpl implements DeviceService
@Autowired
private DeviceSuitService deviceSuitService;
private final String WIFI_MAC = "D8";
private final String FOUR_G_MAC = "4C";
/**
* 查询设备
*
@ -144,8 +148,7 @@ public class DeviceServiceImpl implements DeviceService
* @return 设备
*/
@Override
public List<DeviceVO> selectSmDeviceList(DeviceQuery smDevice)
{
public List<DeviceVO> selectSmDeviceList(DeviceQuery smDevice) {
return deviceMapper.selectSmDeviceList(smDevice);
}
@ -163,23 +166,30 @@ public class DeviceServiceImpl implements DeviceService
String key = data.getDeviceNo();
ServiceUtil.assertion(!redisLock.lock(RedisLockKey.ADD_DEVICE, key), "当前录入人数过多,请稍后再试");
try {
ServiceUtil.assertion(deviceValidator.isRepeatMac(data.getDeviceId(), data.getMac()), "MAC重复");
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为空");
// 创建OneNet设备
int code = iotService.create(data.getMac(), model.getProductId());
ServiceUtil.assertion(!IotHttpStatus.SUCCESS.equalCode(code) && !IotHttpStatus.DEVICE_EXIST.equalCode(code), "设备注册失败");
data.setCreateTime(DateUtils.getNowDate());
data.setStatus(DeviceStatus.NORMAL.getStatus());
Integer result = transactionTemplate.execute(status -> {
// 创建设备
int insert = deviceMapper.insertSmDevice(data);
ServiceUtil.assertion(insert != 1, "新增设备失败");
// 创建OneNet设备1
int code = iotService.create(data.getMac(), model.getProductId());
ServiceUtil.assertion(!IotHttpStatus.SUCCESS.equalCode(code) && !IotHttpStatus.DEVICE_EXIST.equalCode(code), "MAC-1注册失败");
// 创建OneNet设备2
if (StringUtils.hasText(data.getMac2())) {
code = iotService.create(data.getMac2(), model.getProductId());
ServiceUtil.assertion(!IotHttpStatus.SUCCESS.equalCode(code) && !IotHttpStatus.DEVICE_EXIST.equalCode(code), "MAC-2注册失败");
}
return insert;
});
return result == null ? 0 : result;
@ -332,6 +342,16 @@ public class DeviceServiceImpl implements DeviceService
device.setModelId(dto.getModelId());
device.setServiceType(model.getServiceType());
device.setServiceRate(model.getServiceRate());
// 若为双模版本的型号则注册4G或WIFI版本MAC
if (ModelTag.isTwoMac(model.getTags())) {
if (device.getMac().endsWith(FOUR_G_MAC)) {
device.setMac2(StringUtils.replaceLast(device.getMac(), FOUR_G_MAC, WIFI_MAC));
} else if (device.getMac().endsWith(WIFI_MAC)) {
device.setMac2(StringUtils.replaceLast(device.getMac(), WIFI_MAC, FOUR_G_MAC));
}
}
return this.insertSmDevice(device);
}
@ -386,9 +406,9 @@ public class DeviceServiceImpl implements DeviceService
DeviceVO device = selectSmDeviceByDeviceId(deviceId);
ServiceUtil.assertion(device == null, "设备不存在");
if (open) {
return iotService.open(device.getMac(), device.getModelProductId());
return iotService.open(device);
} else {
return iotService.close(device.getMac(), device.getModelProductId());
return iotService.close(device);
}
}
@ -442,10 +462,10 @@ public class DeviceServiceImpl implements DeviceService
// 物联网设备归零
try {
CommandResponse res = iotService.setTime(device.getMac(), 0L, device.getModelProductId());
CommandResponse res = iotService.setTime(device, 0L);
ServiceUtil.assertion( !res.isSuccess(), "设备归零失败,请检查设备是否在线或联系管理员");
boolean close = iotService.close(device.getMac(), device.getModelProductId());
boolean close = iotService.close(device);
ServiceUtil.assertion( !close, "设备关闭失败,请检查设备是否在线或联系管理员");
} catch (Exception e) {
if (required) {
@ -484,7 +504,8 @@ public class DeviceServiceImpl implements DeviceService
// 直接设置电量为0
try {
CommandResponse res = iotService.setEle(device.getMac(), BigDecimal.valueOf(0), device.getModelProductId());
CommandResponse res = iotService.setEle(device, BigDecimal.valueOf(0));
ServiceUtil.assertion(res == null, "归零电量失败,返回值为空");
ServiceUtil.assertion(!res.isSuccess(), "归零电量失败,请检查设备是否在线或联系管理员");
} catch (Exception e) {
if (required) {
@ -512,7 +533,7 @@ public class DeviceServiceImpl implements DeviceService
}
@Override
public int resetEleWithBill(Long deviceId) {
public int resetEleWithBill(Long deviceId, boolean requiredIot) {
if (deviceId == null) {
return 0;
}
@ -537,10 +558,24 @@ public class DeviceServiceImpl implements DeviceService
}
// 发送命令
int reset = this.resetEle(device, true);
int reset = this.resetEle(device, requiredIot);
ServiceUtil.assertion(reset != 1, "设备电量归零失败");
return reset;
return 1;
}
@Override
public boolean addEle(Long deviceId, BigDecimal ele, boolean withIot, String reason) {
DeviceVO device = selectSmDeviceByDeviceId(deviceId);
if (device == null) {
return false;
}
if (withIot) {
CommandResponse res = iotService.addEle(device, ele);
return res.isSuccess();
}
return true;
}
@ -566,7 +601,7 @@ public class DeviceServiceImpl implements DeviceService
DeviceVO newDevice = selectSmDeviceByDeviceId(deviceId);
long betweenSeconds = Duration.between(LocalDateTime.now(), newDevice.getExpireTime()).getSeconds();
if (betweenSeconds > 0) {
CommandResponse rechargeResult = iotService.setTime(device.getMac(), betweenSeconds, device.getModelProductId());
CommandResponse rechargeResult = iotService.setTime(device, betweenSeconds);
ServiceUtil.assertion(!rechargeResult.isSuccess(), "设备充值失败,请检查设备是否在线");
}
}
@ -630,11 +665,16 @@ public class DeviceServiceImpl implements DeviceService
log.info("device list is empty");
return;
}
LocalDateTime now = LocalDateTime.now();
for (DeviceVO device : list) {
if (StringUtils.hasText(device.getMac())) {
String status = iotService.getOnlineStatus(device.getMac(), device.getModelProductId()).getStatus();
String status = iotService.getOnlineStatus(device).getStatus();
// log.info("device: {} {} online status is {}", device.getDeviceId(), device.getMac(), status);
device.setOnlineStatus(status);
if (DeviceOnlineStatus.ONLINE.getStatus().equals(status)) {
device.setLastOnlineTime(now);
}
} else {
device.setOnlineStatus(DeviceOnlineStatus.OFFLINE.getStatus());
log.info("device {} mac is empty", device.getDeviceId());
@ -685,9 +725,9 @@ public class DeviceServiceImpl implements DeviceService
ServiceUtil.assertion(!UserUtil.hasFrontUser(device.getUserId()), "该设备不是您的,无法进行该操作" );
if (DevicePowerStatus.ON.equals(status)) {
iotService.open(device.getMac(), device.getModelProductId());
iotService.open(device);
} else if (DevicePowerStatus.OFF.equals(status)) {
iotService.close(device.getMac(), device.getModelProductId());
iotService.close(device);
} else {
throw new ServiceException("不支持的操作");
}
@ -727,7 +767,7 @@ public class DeviceServiceImpl implements DeviceService
continue;
}
// 获取数据点信息
IotDeviceInfo deviceInfo = iotService.getDeviceInfo(device.getMac(), device.getModelProductId());
IotDeviceInfo deviceInfo = iotService.getDeviceInfo(device);
// 更新设备信息
Device data = new Device();
@ -737,6 +777,9 @@ public class DeviceServiceImpl implements DeviceService
data.setTotalElectriQuantity(deviceInfo.getW());
data.setPowerStatus(deviceInfo.getS());
data.setRemainTime(deviceInfo.getTime());
data.setRealTimePower(deviceInfo.getP());
data.setVoltage(deviceInfo.getV());
data.setElectricity(deviceInfo.getA());
// 是否有WIFI
if (ModelTag.hasTag(device.getModelTags(), ModelTag.WIFI)) {
@ -765,7 +808,11 @@ public class DeviceServiceImpl implements DeviceService
}
// 是否在线
data.setOnlineStatus(iotService.getOnlineStatus(device.getMac(), device.getModelProductId()).getStatus());
String onlineStatus = iotService.getOnlineStatus(device).getStatus();
data.setOnlineStatus(onlineStatus);
if (DeviceOnlineStatus.ONLINE.getStatus().equals(onlineStatus)) {
device.setLastOnlineTime(now);
}
deviceMapper.updateSmDevice(data);
} catch (Exception e) {
@ -832,14 +879,21 @@ public class DeviceServiceImpl implements DeviceService
ServiceUtil.assertion(!Objects.equals(store.getUserId(), userId), "该店铺不属于该用户");
}
// 绑定
int updateCount = deviceMapper.bindStore(device.getDeviceId(), storeId, userId);
ServiceUtil.assertion(updateCount != 1, "当前设备信息已变更,请刷新后重试");
transactionTemplate.execute(status -> {
// 绑定
int updateCount = deviceMapper.bind(device.getDeviceId(), storeId, userId);
ServiceUtil.assertion(updateCount != 1, "当前设备信息已变更,请刷新后重试");
// 设备绑定记录
scheduledExecutorService.schedule(() -> {
smDeviceBindRecordService.record(userId, device.getDeviceId());
}, 0, TimeUnit.SECONDS);
// 记录绑定记录
int record = smDeviceBindRecordService.record(userId, device.getDeviceId());
ServiceUtil.assertion(record != 1, "添加绑定记录失败");
// 用户设置为商户
boolean changeType = userService.changeType(userId, UserType.MCH);
ServiceUtil.assertion(record != 1, "修改用户类型失败");
return updateCount;
});
return device.getMac();
}
@ -858,7 +912,7 @@ public class DeviceServiceImpl implements DeviceService
}
@Override
public int resetTimeWithBill(Long deviceId) {
public int resetTimeWithBill(Long deviceId, boolean requiredIot) {
ServiceUtil.assertion(deviceId == null, "设备id不能为空");
// 拉取最新设备信息
@ -884,10 +938,10 @@ public class DeviceServiceImpl implements DeviceService
}
// 发送命令
int reset = this.resetTime(device, true);
int reset = this.resetTime(device, requiredIot);
ServiceUtil.assertion(reset != 1, "归零失败");
return reset;
return 1;
}
/**

View File

@ -147,7 +147,17 @@ public class DeviceValidatorImpl extends BaseValidator implements DeviceValidato
if (StringUtils.isBlank(mac)) {
return false;
}
DeviceVO device = deviceService.selectByMac(mac);
return device != null && !Objects.equals(deviceId, device.getDeviceId());
DeviceQuery query = new DeviceQuery();
query.setAnyMac(mac);
List<DeviceVO> repeatList = deviceService.selectSmDeviceList(query);
if (CollectionUtils.isNotEmptyElement(repeatList)) {
for (DeviceVO device : repeatList) {
boolean repeat = device != null && !Objects.equals(deviceId, device.getDeviceId());
if (repeat) {
return true;
}
}
}
return false;
}
}

View File

@ -148,7 +148,7 @@ public class SmMeterReadingRecordServiceImpl implements ISmMeterReadingRecordSer
if (StringUtils.isAnyBlank(device.getMac(), device.getModelProductId())) {
continue;
}
IotDeviceInfo deviceInfo = iotService.getDeviceInfo(device.getMac(), device.getModelProductId());
IotDeviceInfo deviceInfo = iotService.getDeviceInfo(device);
SmMeterReadingRecordVo history = recordHistory.stream().filter(item -> Objects.equals(device.getDeviceId(), item.getDeviceId())).findFirst().orElse(null);
// 抄表记录

View File

@ -11,6 +11,11 @@ import java.util.List;
*/
@Data
public class SmModelQuery extends SmModel {
@ApiModelProperty("型号id列表")
private List<Long> modelIds;
@ApiModelProperty("关键词")
private String keyword;
}

View File

@ -47,4 +47,20 @@ public enum ModelTag {
public static boolean hasTag(List<String> modelTags, ModelTag modelTag) {
return modelTags.contains(modelTag.getTag());
}
/**
* 获取列表
*/
public static List<String> asList(ModelTag ...tags) {
return Arrays.stream(tags).map(ModelTag::getTag).collect(Collectors.toList());
}
/**
* 判断是否为双模的型号功能
*/
public static boolean isTwoMac(List<String> tags) {
// 双模
List<String> towMacTags = asList(FOUR_G, WIFI);
return new HashSet<>(towMacTags).containsAll(tags);
}
}

View File

@ -37,6 +37,12 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="productId != null and productId != ''"> and sm.product_id = #{productId}</if>
<if test="deleted == null">and sm.deleted = false</if>
<if test="deleted != null">and sm.deleted = #{deleted}</if>
<if test="keyword != null">
and (
sm.model_name like concat('%', #{keyword}, '%')
or sm.model like concat('%', #{keyword}, '%')
)
</if>
</sql>
<select id="selectSmModelList" parameterType="SmModelQuery" resultMap="SmModelResult">

View File

@ -279,38 +279,41 @@ public class PayBillServiceImpl implements PayBillService
ServiceUtil.assertion(payBill == null, "支付订单不存在");
Long lockKey = payBill.getPayId();
ServiceUtil.assertion(redisLock.lock(RedisLockKey.PAY_BILL_SUCCESS, lockKey), "支付订单正在处理中:payId=" + payBill.getPayId());
// ServiceUtil.assertion(redisLock.lock(RedisLockKey.PAY_BILL_SUCCESS, lockKey), "支付订单正在处理中:payId=" + payBill.getPayId());
try {
// 若已经支付成功则跳过
if (PayBillStatus.payedList().contains(payBill.getStatus())) {
return;
// 若不是支付成功的状态则修改为支付成功
if (!PayBillStatus.payedList().contains(payBill.getStatus())) {
ServiceUtil.assertion(!PayBillStatus.PAYING.getStatus().equals(payBill.getStatus()), "该支付订单不是正在支付的支付订单");
Integer result = transactionTemplate.execute(status -> {
// 修改支付订单状态
PayBill data = new PayBill();
data.setStatus(PayBillStatus.PAY_SUCCESS.getStatus());
data.setPayTime(payTime);
PayBillQuery query = new PayBillQuery();
query.setStatus(PayBillStatus.PAYING.getStatus());
query.setPayId(payBill.getPayId());
int update = this.updateByQuery(data, query);
ServiceUtil.assertion(update != 1, "支付订单状态已改变,请稍后再试");
return update;
});
}
ServiceUtil.assertion(!PayBillStatus.PAYING.getStatus().equals(payBill.getStatus()), "该支付订单不是正在支付的支付订单");
transactionTemplate.execute(status -> {
// 修改支付订单状态
PayBill data = new PayBill();
data.setStatus(PayBillStatus.PAY_SUCCESS.getStatus());
data.setPayTime(payTime);
PayBillQuery query = new PayBillQuery();
query.setStatus(PayBillStatus.PAYING.getStatus());
query.setPayId(payBill.getPayId());
int update = this.updateByQuery(data, query);
ServiceUtil.assertion(update != 1, "支付订单状态已改变,请稍后再试");
// 处理业务
// 获取修改后的数据
PayBillVO bill = this.selectPayBillByPayId(payBill.getPayId());
// 若为成功则处理业务
if (PayBillStatus.PAY_SUCCESS.getStatus().equals(bill.getStatus())) {
PayBillBstType bstType = PayBillBstType.parse(payBill.getBstType());
if (bstType != null && bstType.getAfterPay() != null) {
PayBillVO newPayBill = selectPayBillByPayId(payBill.getPayId());
AfterPay afterPay = SpringUtils.getBean(bstType.getAfterPay());
int bstResult = afterPay.onPaySuccess(newPayBill);
ServiceUtil.assertion(bstResult == 0, "业务处理失败");
}
return update;
});
};
}
} finally {
redisLock.unlock(RedisLockKey.PAY_BILL_SUCCESS, lockKey);
// redisLock.unlock(RedisLockKey.PAY_BILL_SUCCESS, lockKey);
}
}

View File

@ -174,7 +174,7 @@ public class TimeBillServiceImpl implements TimeBillService, AfterPay
ServiceUtil.assertion(insert != 1, "下单失败,请稍后再试");
// 开启设备
boolean open = iotService.open(device.getMac(), device.getModelProductId());
boolean open = iotService.open(device);
ServiceUtil.assertion(!open, "开启设备失败,请稍后再试");
return insert;
@ -219,7 +219,7 @@ public class TimeBillServiceImpl implements TimeBillService, AfterPay
// 关闭设备
DeviceVO device = deviceService.selectSmDeviceByDeviceId(bill.getDeviceId());
ServiceUtil.assertion(device == null, "设备不存在");
boolean close = iotService.close(device.getMac(), device.getModelProductId());
boolean close = iotService.close(device);
ServiceUtil.assertion(!close, "关闭设备失败,请检查设备是否在线");
return update;

View File

@ -218,6 +218,11 @@ public class TransactionBill extends BaseEntity implements Payable
@JsonView(JsonViewProfile.App.class)
private String deviceMac;
@Excel(name = "设备MAC2")
@ApiModelProperty("设备MAC2")
@JsonView(JsonViewProfile.App.class)
private String deviceMac2;
@ApiModelProperty("总计退款金额")
@JsonView(JsonViewProfile.App.class)
private BigDecimal refundAmount;

View File

@ -2,6 +2,7 @@ package com.ruoyi.ss.transactionBill.domain.vo;
import com.fasterxml.jackson.annotation.JsonView;
import com.ruoyi.common.core.domain.JsonViewProfile;
import com.ruoyi.iot.interfaces.IotDevice;
import com.ruoyi.ss.suit.domain.enums.SuitTimeUnit;
import com.ruoyi.ss.transactionBill.domain.TransactionBill;
import io.swagger.annotations.ApiModel;
@ -16,7 +17,7 @@ import java.math.BigDecimal;
*/
@ApiModel
@Data
public class TransactionBillVO extends TransactionBill {
public class TransactionBillVO extends TransactionBill implements IotDevice {
@ApiModelProperty("用户名称")
@JsonView(JsonViewProfile.App.class)
private String userName;
@ -65,4 +66,18 @@ public class TransactionBillVO extends TransactionBill {
return time * unit.getConversion();
}
@Override
public String getMac1() {
return getDeviceMac();
}
@Override
public String getMac2() {
return getDeviceMac2();
}
@Override
public String getProductId() {
return getDeviceProductId();
}
}

View File

@ -53,6 +53,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
stb.device_no,
stb.suit_name,
stb.device_mac,
stb.device_mac2,
stb.refund_amount,
stb.refund_mch_amount,
stb.refund_service_amount,
@ -152,6 +153,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="query.mchName != null "> and su1.user_name like concat('%', #{query.mchName}, '%')</if>
<if test="query.deviceName != null "> and stb.device_name like concat('%', #{query.deviceName}, '%')</if>
<if test="query.deviceMac != null and query.deviceMac != ''"> and stb.device_mac = #{query.deviceMac}</if>
<if test="query.deviceMac2 != null and query.deviceMac2 != ''"> and stb.device_mac2 = #{query.deviceMac2}</if>
<if test="query.createTime != null"> and stb.create_time = #{query.createTime}</if>
<if test="query.createDate != null"> and date(stb.create_time) = date(#{query.createDate})</if>
<if test="query.year != null "> and year(stb.create_time) = #{query.year}</if>
@ -433,6 +435,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="deviceNo != null ">device_no,</if>
<if test="suitName != null ">suit_name,</if>
<if test="deviceMac != null ">device_mac,</if>
<if test="deviceMac2 != null">device_mac2,</if>
<if test="refundAmount != null">refund_amount,</if>
<if test="refundMchAmount != null">refund_mch_amount,</if>
<if test="refundServiceAmount != null">refund_service_amount,</if>
@ -486,6 +489,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="deviceNo != null ">#{deviceNo},</if>
<if test="suitName != null ">#{suitName},</if>
<if test="deviceMac != null ">#{deviceMac},</if>
<if test="deviceMac2 != null">#{deviceMac2},</if>
<if test="refundAmount != null">#{refundAmount},</if>
<if test="refundMchAmount != null">#{refundMchAmount},</if>
<if test="refundServiceAmount != null">#{refundServiceAmount},</if>
@ -517,7 +521,6 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
where stb.bill_id = #{data.billId}
</update>
<!--FIXME 更新两个表都有的字段时,会报错-->
<sql id="updateColumns">
<if test="data.userId != null">stb.user_id = #{data.userId},</if>
<if test="data.type != null">stb.`type` = #{data.type},</if>
@ -557,6 +560,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="data.deviceNo != null ">stb.device_no = #{data.deviceNo},</if>
<if test="data.suitName != null ">stb.suit_name = #{data.suitName},</if>
<if test="data.deviceMac != null ">stb.device_mac = #{data.deviceMac},</if>
<if test="data.deviceMac2 != null">stb.device_mac2 = #{data.deviceMac2},</if>
<if test="data.refundAmount != null">stb.refund_amount = #{data.refundAmount},</if>
<if test="data.refundMchAmount != null">stb.refund_mch_amount = #{data.refundMchAmount},</if>
<if test="data.refundServiceAmount != null">stb.refund_service_amount = #{data.refundServiceAmount},</if>
@ -568,6 +572,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<if test="data.suitEnableLowPowerClose != null">suit_enable_low_power_close = #{data.suitEnableLowPowerClose},</if>
<if test="data.suitLowPower != null">suit_low_power = #{data.suitLowPower},</if>
<if test="data.deviceProductId != null">device_product_id = #{data.deviceProductId},</if>
<if test="data.deviceRechargeStatus != null">device_recharge_status = #{data.deviceRechargeStatus},</if>
</sql>
<update id="updateByQuery">
@ -663,7 +668,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
<update id="updateDeviceRechargeStatus">
update sm_transaction_bill
set device_recharge_status = #{status}
where bill_id = #{billId} and `type` = '1' and device_recharge_status != '1'
where bill_id = #{billId} and `type` = '1' and device_recharge_status not in('1', '3')
</update>
<update id="updateDeviceRechargeStatusByIds">

View File

@ -65,7 +65,7 @@ public class RechargeDepositAfterPay implements AfterPay {
ServiceUtil.assertion(update != 1, "修改订单信息失败,状态已经发生改变");
try {
iotService.open(device.getMac(), device.getModelProductId());
iotService.open(device);
} catch (Exception e) {
log.error(e.getMessage());
}

View File

@ -2,11 +2,8 @@ package com.ruoyi.ss.transactionBill.service.impl;
import com.github.pagehelper.PageHelper;
import com.ruoyi.common.constant.Constants;
import com.ruoyi.common.constant.HttpStatus;
import com.ruoyi.common.core.domain.ValidateResult;
import com.ruoyi.common.core.redis.RedisLock;
import com.ruoyi.common.core.redis.enums.RedisLockKey;
import com.ruoyi.common.enums.BusinessStatus;
import com.ruoyi.common.enums.WithdrawServiceType;
import com.ruoyi.common.exception.ServiceException;
import com.ruoyi.common.utils.*;
@ -390,6 +387,7 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
order.setMchId(mch.getUserId());
order.setDeviceName(device.getDeviceName());
order.setDeviceMac(device.getMac());
order.setDeviceMac2(device.getMac2());
order.setDeviceProductId(device.getModelProductId());
// 店铺信息
@ -753,7 +751,7 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
@Override
public boolean rechargeDevice(Long billId) {
ServiceUtil.assertion(billId == null, "参数错误,billId不允许为空");
ServiceUtil.assertion(!redisLock.lock(RedisLockKey.RECHARGE_DEVICE, billId), "当前设备充值请求过于频繁,请等待");
// ServiceUtil.assertion(!redisLock.lock(RedisLockKey.RECHARGE_DEVICE, billId), "当前设备充值请求过于频繁,请等待");
try {
TransactionBillVO bill = transactionBillMapper.selectSmTransactionBillByBillId(billId);
ServiceUtil.assertion(bill == null || !TransactionBillType.RECHARGE.getType().equals(bill.getType()), "不存在的充值订单");
@ -796,7 +794,8 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
openTime = openTime.add(device.getSurplusEle().abs());
}
// 设备充值电量
CommandResponse res = iotService.addEle(device.getMac(), openTime, device.getModelProductId());
CommandResponse res = iotService.addEle(device, openTime);
ServiceUtil.assertion(res == null, "设备充值失败,返回数据为空");
ServiceUtil.assertion(!res.isSuccess(), "设备充值失败:" + res.getMsg());
return true;
} else {
@ -822,7 +821,7 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
return success;
} finally {
redisLock.unlock(RedisLockKey.RECHARGE_DEVICE, billId);
// redisLock.unlock(RedisLockKey.RECHARGE_DEVICE, billId);
}
}
@ -889,9 +888,9 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
return updateCount;
});
// 异步充值设备尝试3
// 充值设备尝试1
if (result != null && result == 1) {
this.tryRechargeDevice(bill.getBillId(), 3);
this.tryRechargeDevice(bill.getBillId(), 1);
}
return result == null ? 0 : result;
@ -906,14 +905,14 @@ 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);
// 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
@ -973,7 +972,7 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
} else if(SuitFeeType.TIMING_TIME.getType().equals(order.getSuitFeeType())){
deviceService.resetTime(device, true);
}
iotService.close(device.getMac(), device.getModelProductId());
iotService.close(device);
} catch (Exception e) {
log.error("尝试关闭设备失败");
}
@ -1367,6 +1366,7 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
dto.setStatus(TransactionBillStatus.SUCCESS.getStatus());
dto.setDeviceRechargeStatus(TransactionBillDeviceRechargeStatus.FAIL.getStatus());
dto.setType(TransactionBillType.RECHARGE.getType());
dto.setIsFinished(false);
List<TransactionBillVO> list = this.selectSmTransactionBillList(dto);
if (!CollectionUtils.isEmptyElement(list)) {
@ -1450,10 +1450,17 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
// 拉取设备信息
deviceService.pullDeviceInfo(bill.getDeviceId());
Boolean execute = transactionTemplate.execute(status -> {
Integer execute = transactionTemplate.execute(status -> {
// 更新设备充值状态
boolean result = transactionBillMapper.bluetoothRechargeSuccess(billNo) == 1;
ServiceUtil.assertion(!result, "蓝牙充值回调失败");
TransactionBill data = new TransactionBill();
data.setDeviceRechargeStatus(TransactionBillDeviceRechargeStatus.SUCCESS.getStatus());
TransactionBillQuery query = new TransactionBillQuery();
query.setBillId(bill.getBillId());
query.setType(TransactionBillType.RECHARGE.getType());
query.setDeviceRechargeStatus(TransactionBillDeviceRechargeStatus.BLUETOOTH.getStatus());
query.setStatus(TransactionBillStatus.SUCCESS.getStatus());
int result = this.updateByQuery(data, query);
ServiceUtil.assertion( result != 1, "蓝牙充值回调失败,设备状态已发生改变");
// 更新套餐使用信息
DeviceVO device = deviceService.selectSmDeviceByDeviceId(bill.getDeviceId());
@ -1468,7 +1475,7 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
});
// 是否成功
boolean success = execute != null && execute;
boolean success = execute != null && execute == 1;
// 成功后操作
if (success) {
// 时长变化记录
@ -1678,9 +1685,9 @@ public class TransactionBillServiceImpl implements TransactionBillService, After
ServiceUtil.assertion(StringUtils.isBlank(device.getMac()), "设备MAC为空请联系管理员处理");
if (open) {
return iotService.open(device.getMac(), device.getModelProductId()) ? 1 : 0;
return iotService.open(device) ? 1 : 0;
} else {
return iotService.close(device.getMac(), device.getModelProductId()) ? 1 : 0;
return iotService.close(device) ? 1 : 0;
}
}

View File

@ -0,0 +1,24 @@
package com.ruoyi.ss.user.domain.dto;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.Size;
/**
* @author wjh
* 2024/9/25
*/
@Data
public class UserUpdatePasswordDTO {
@ApiModelProperty("旧密码")
private String oldPassword;
@ApiModelProperty("新密码")
@NotBlank(message = "新密码不能为空")
@Size(min = 6, max = 32, message = "新密码长度应该在6~32个字符之间")
private String newPassword;
}

View File

@ -481,6 +481,9 @@ public class SmUserServiceImpl implements ISmUserService
public int insertSmUser(SmUser smUser)
{
this.validate(smUser, false);
if (StringUtils.hasText(smUser.getPassword())) {
smUser.setPassword(SecurityUtils.encryptPassword(smUser.getPassword()));
}
smUser.setCreateTime(DateUtils.getNowDate());
return smUserMapper.insertSmUser(smUser);
}
@ -525,6 +528,9 @@ public class SmUserServiceImpl implements ISmUserService
public int updateSmUser(SmUser smUser)
{
this.validate(smUser, true);
if (StringUtils.hasText(smUser.getPassword())) {
smUser.setPassword(SecurityUtils.encryptPassword(smUser.getPassword()));
}
smUser.setUpdateTime(DateUtils.getNowDate());
return smUserMapper.updateSmUser(smUser);
}

View File

@ -67,13 +67,13 @@ public class BillLowPowerTask {
}
try {
// 判断设备是否在线
DeviceOnlineStatus online = iotService.getOnlineStatus(bill.getDeviceMac(), bill.getDeviceProductId());
DeviceOnlineStatus online = iotService.getOnlineStatus(bill);
if (!DeviceOnlineStatus.ONLINE.equals(online)) {
log.warn("关闭低功率订单{}时,设备未在线", bill.getBillNo());
continue;
}
IotDeviceInfo deviceInfo = iotService.getDeviceInfo(bill.getDeviceMac(), bill.getDeviceProductId());
IotDeviceInfo deviceInfo = iotService.getDeviceInfo(bill);
// 判断是否低功率若低于指定功率则关闭订单
if (deviceInfo != null && deviceInfo.getP() != null && deviceInfo.getP().compareTo(bill.getSuitLowPower()) < 0) {
EndUseDTO dto = new EndUseDTO();

View File

@ -23,6 +23,8 @@ import com.ruoyi.ss.device.service.DeviceValidator;
import com.ruoyi.ss.device.service.DeviceService;
import com.ruoyi.ss.meterReadingRecord.domain.SmMeterReadingRecordQuery;
import com.ruoyi.ss.meterReadingRecord.service.ISmMeterReadingRecordService;
import com.ruoyi.ss.suit.domain.enums.SuitFeeType;
import com.ruoyi.ss.suit.domain.enums.SuitTimeUnit;
import com.ruoyi.web.core.annotation.DeviceAdminRequired;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
@ -31,6 +33,7 @@ import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.*;
import java.math.BigDecimal;
import java.util.Collections;
import java.util.List;
@ -128,9 +131,9 @@ public class AppDeviceController extends BaseController {
@Log(title = "设备归零", businessType = BusinessType.OTHER, operatorType = OperatorType.MOBILE)
@ApiOperation("设备归零")
@PutMapping("{deviceId}/reset")
public AjaxResult reset(@PathVariable @ApiParam("设备id") Long deviceId) {
int r1 = smDeviceService.resetTimeWithBill(deviceId);
int r2 = smDeviceService.resetEleWithBill(deviceId);
public AjaxResult reset(@PathVariable @ApiParam("设备id") Long deviceId, @RequestParam(required = false, defaultValue = "true") Boolean requiredIot) {
int r1 = smDeviceService.resetTimeWithBill(deviceId, requiredIot);
int r2 = smDeviceService.resetEleWithBill(deviceId, requiredIot);
return toAjax(r1 >= 0 && r2 >= 0);
}
@ -141,15 +144,31 @@ public class AppDeviceController extends BaseController {
return success(smMeterReadingRecordService.selectCount(dto));
}
@Log(title = "设备充值时长", businessType = BusinessType.OTHER, operatorType = OperatorType.MOBILE)
@ApiOperation("设备充值时长")
@Log(title = "商户充值设备", businessType = BusinessType.OTHER, operatorType = OperatorType.MOBILE)
@ApiOperation("商户充值设备")
@PutMapping("/addTime/{deviceId}")
public AjaxResult addTime(@PathVariable @ApiParam("设备id") Long deviceId, @ApiParam("时长(分钟)") Long amount)
{
public AjaxResult addTime(@PathVariable @ApiParam("设备id") Long deviceId,
@ApiParam("数值") @RequestParam BigDecimal amount,
@ApiParam("单位") @RequestParam(required = false, defaultValue = "3") String timeUnit,
@ApiParam("是否操作物联网设备") @RequestParam(required = false, defaultValue = "true") Boolean withIot
) {
ServiceUtil.assertion(!deviceValidator.isBelong(deviceId, getUserId()), "这不是您的设备");
DeviceVO device = smDeviceService.selectSmDeviceByDeviceId(deviceId);
ServiceUtil.assertion(device == null || !getUserId().equals(device.getUserId()), "设备不存在或您无权充值");
return toAjax(smDeviceService.addTimeByUser(deviceId, amount * 60, true, "商户手动充值"));
// 电量
if ( "0".equals(timeUnit)) {
return toAjax(smDeviceService.addEle(deviceId, amount, withIot, "商户手动充值"));
}
// 时长
else {
SuitTimeUnit unit = SuitTimeUnit.getByValue(timeUnit);
if (unit == null) {
return error("非法的时长单位");
}
long seconds = amount.multiply(new BigDecimal(unit.getConversion())).longValue();
return toAjax(smDeviceService.addTimeByUser(deviceId, seconds, withIot, "商户手动充值"));
}
}
@ApiOperation("刷新设备信息")

View File

@ -354,4 +354,14 @@ public class AppTransactionBillController extends BaseController
return success(transactionBillService.pay(bo));
}
@ApiOperation("充值订单设备")
@PutMapping("/rechargeBillDevice")
public AjaxResult rechargeBillDevice(@RequestParam String billNo) {
TransactionBillVO bill = transactionBillService.selectSmTransactionBillByBillNo(billNo);
if (bill == null) {
return error("订单不存在");
}
return success(transactionBillService.rechargeDevice(bill.getBillId()));
}
}

View File

@ -5,12 +5,18 @@ import com.ruoyi.common.constant.HttpStatus;
import com.ruoyi.common.core.controller.BaseController;
import com.ruoyi.common.core.domain.AjaxResult;
import com.ruoyi.common.core.domain.JsonViewProfile;
import com.ruoyi.common.core.domain.model.LoginUser;
import com.ruoyi.common.enums.UserType;
import com.ruoyi.common.exception.ServiceException;
import com.ruoyi.common.utils.SecurityUtils;
import com.ruoyi.common.utils.ServiceUtil;
import com.ruoyi.common.utils.StringUtils;
import com.ruoyi.framework.web.service.SysPasswordService;
import com.ruoyi.framework.web.service.TokenService;
import com.ruoyi.ss.user.domain.bo.SmUserBO;
import com.ruoyi.ss.user.domain.SmUserVo;
import com.ruoyi.ss.user.domain.dto.UserRealNameDTO;
import com.ruoyi.ss.user.domain.dto.UserUpdatePasswordDTO;
import com.ruoyi.ss.user.service.ISmUserService;
import com.ruoyi.ss.user.service.UserAssembler;
import com.ruoyi.system.domain.enums.verificationCode.CodeBusinessType;
@ -43,6 +49,12 @@ public class AppUserController extends BaseController {
@Autowired
private UserAssembler userAssembler;
@Autowired
private SysPasswordService passwordService;
@Autowired
private TokenService tokenService;
@ApiOperation("获取当前登录前台用户的信息")
@GetMapping("/userInfo")
@JsonView(JsonViewProfile.AppMch.class)
@ -64,12 +76,28 @@ public class AppUserController extends BaseController {
return AjaxResult.success(userService.changeType(getUserId(), UserType.parse(userType)));
}
@ApiOperation("修改密码")
@ApiOperation("使用旧密码修改密码")
@PutMapping("/updatePassword")
public AjaxResult updatePassword(SmUserBO bo) {
ServiceUtil.assertion(!verificationCodeService.checkCode(bo.getPhonenumber(), bo.getMobileCode(), CodeBusinessType.UPDATE_PASSWORD, true),
"短信验证码错误,请重试");
return AjaxResult.success(userService.updatePassword(getUserId(), bo.getPassword()));
public AjaxResult updatePassword( @RequestBody @Validated UserUpdatePasswordDTO dto) {
SmUserVo user = userService.selectSmUserByUserId(getUserId());
if (user == null) {
return error("用户不存在");
}
// 若有设置旧密码则应该判断旧密码是否正确
if (StringUtils.hasText(user.getPassword()) && !passwordService.matches(user, dto.getOldPassword())) {
return error("旧密码错误");
}
return toAjax(userService.updatePassword(getUserId(), dto.getNewPassword()));
}
@ApiOperation("退出登录")
@DeleteMapping("/logout")
public AjaxResult logout() {
// 删除用户缓存记录
LoginUser loginUser = SecurityUtils.getLoginUser();
tokenService.delLoginUser(loginUser.getToken());
return success();
}
@ApiOperation("用户实名认证")

View File

@ -58,6 +58,7 @@ public class SmUserController extends BaseController
List<SmUserVo> list = smUserService.selectSmUserList(smUser);
userAssembler.assembleStoreCount(list);
userAssembler.assembleDeviceCount(list);
smUserService.desensitization(list);
return getDataTable(list);
}
@ -96,6 +97,7 @@ public class SmUserController extends BaseController
List<SmUserVo> list = Collections.singletonList(user);
userAssembler.assembleStoreCount(list);
userAssembler.assembleDeviceCount(list);
smUserService.desensitization(list);
return success(user);
}