双模设备、用户密码、蓝牙设备充值、支付回调优化
This commit is contained in:
parent
49469ba6f1
commit
e2b9f18a21
|
@ -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;
|
||||
|
|
|
@ -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)));
|
||||
}
|
||||
}
|
||||
|
|
|
@ -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", "*/*");
|
||||
|
|
|
@ -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));
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -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();
|
||||
|
||||
}
|
|
@ -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());
|
||||
|
|
|
@ -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);
|
||||
}
|
||||
|
|
|
@ -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;
|
||||
}
|
||||
|
|
|
@ -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;
|
||||
}
|
||||
|
|
|
@ -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;
|
||||
}
|
||||
|
|
|
@ -10,6 +10,7 @@ import lombok.Data;
|
|||
public class DeviceMacSnVO {
|
||||
|
||||
private String mac;
|
||||
private String mac2;
|
||||
private String sn;
|
||||
|
||||
}
|
||||
|
|
|
@ -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();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -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);
|
||||
|
||||
/**
|
||||
* 解绑商户
|
||||
|
|
|
@ -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=")">
|
||||
|
|
|
@ -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);
|
||||
}
|
||||
|
|
|
@ -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;
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -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;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -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);
|
||||
|
||||
// 抄表记录
|
||||
|
|
|
@ -11,6 +11,11 @@ import java.util.List;
|
|||
*/
|
||||
@Data
|
||||
public class SmModelQuery extends SmModel {
|
||||
|
||||
@ApiModelProperty("型号id列表")
|
||||
private List<Long> modelIds;
|
||||
|
||||
@ApiModelProperty("关键词")
|
||||
private String keyword;
|
||||
|
||||
}
|
||||
|
|
|
@ -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);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -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">
|
||||
|
|
|
@ -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);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -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;
|
||||
|
|
|
@ -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;
|
||||
|
|
|
@ -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();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -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">
|
||||
|
|
|
@ -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());
|
||||
}
|
||||
|
|
|
@ -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;
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -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;
|
||||
|
||||
}
|
|
@ -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);
|
||||
}
|
||||
|
|
|
@ -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();
|
||||
|
|
|
@ -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("刷新设备信息")
|
||||
|
|
|
@ -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()));
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -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("用户实名认证")
|
||||
|
|
|
@ -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);
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue
Block a user