|
@@ -2,6 +2,7 @@ package com.gyee.edge.gdnxak.iec102;
|
|
|
|
|
|
import com.gyee.edge.gdnxak.forcast.ForcastModel;
|
|
|
import com.gyee.edge.gdnxak.forcast.ForcastService;
|
|
|
+import com.gyee.edge.gdnxak.utils.ByteUtil;
|
|
|
import com.gyee.edge.gdnxak.utils.SpringContextUtil;
|
|
|
import io.netty.bootstrap.Bootstrap;
|
|
|
import io.netty.channel.Channel;
|
|
@@ -12,7 +13,9 @@ import io.netty.channel.nio.NioEventLoopGroup;
|
|
|
import io.netty.channel.socket.nio.NioSocketChannel;
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
|
|
|
+import java.io.UnsupportedEncodingException;
|
|
|
import java.util.Date;
|
|
|
+import java.util.HashMap;
|
|
|
|
|
|
@Slf4j
|
|
|
public class Iec102Client {
|
|
@@ -29,6 +32,10 @@ public class Iec102Client {
|
|
|
/** 将<code>Channel</code>保存起来, 可用于在其他非handler的地方发送数据 */
|
|
|
private Channel channel;
|
|
|
|
|
|
+ //长文件缓存,待文件全部传输完毕后,再解析转发;例如:LLCDQ, DQ等
|
|
|
+ private Iec102Message longFileMessage;
|
|
|
+ private HashMap<String, Iec102Message> msgMap;
|
|
|
+
|
|
|
public Iec102Client(String host, int port) {
|
|
|
this.host = host;
|
|
|
this.port = port;
|
|
@@ -103,11 +110,18 @@ public class Iec102Client {
|
|
|
sendMessage(msg);
|
|
|
}
|
|
|
|
|
|
- public void callUserData() {
|
|
|
+ //todo: 多文件传输时,注意FCB位置的重置,即5a、7a交替发送
|
|
|
+ public void callUserData(int frameCnt) {
|
|
|
+ Iec102Message iec102Message = null;
|
|
|
+ log.info("framecnt = " + frameCnt);
|
|
|
//召唤一级数据
|
|
|
- Iec102Message msg23 = BasicInstruction102.createFixFrameMessage(BasicInstruction102.C_CALL_USER1);
|
|
|
+ if (0 == frameCnt%2){
|
|
|
+ iec102Message = BasicInstruction102.createFixFrameMessage(BasicInstruction102.C_CALL_USER11);
|
|
|
+ } else {
|
|
|
+ iec102Message = BasicInstruction102.createFixFrameMessage(BasicInstruction102.C_CALL_USER1);
|
|
|
+ }
|
|
|
// Iec102Message msg23 = BasicInstruction102.createFixFrameMessage(BasicInstruction102.M_USERDATA2);
|
|
|
- sendMessage(msg23);
|
|
|
+ sendMessage(iec102Message);
|
|
|
}
|
|
|
|
|
|
public void close() {
|
|
@@ -130,16 +144,34 @@ public class Iec102Client {
|
|
|
break;
|
|
|
}
|
|
|
} else if (Iec102FrameType.Mutable == response.getFrameType()) {
|
|
|
+
|
|
|
+ Iec102Message iMsg = checkFileMessage(response);
|
|
|
//处理用户数据
|
|
|
- log.info("收到用户消息,文件名:" + response.getDataFile());
|
|
|
- log.info("文件内容:" + response.getDataContent());
|
|
|
- //todo: 解析光功率预测文件
|
|
|
- ForcastModel forcastModel = forcastService.readForcastFileString(response.getDataContent());
|
|
|
- forcastService.putForcastModel(forcastModel);
|
|
|
- if (forcastModel != null) {
|
|
|
- log.info(forcastModel.dataType + ", size = " + forcastModel.getForcastData().size());
|
|
|
+ log.info("收到用户消息,文件名:" + iMsg.getDataFile());
|
|
|
+
|
|
|
+ if (0x08 == response.getCOT()) {
|
|
|
+ //再次发送召唤消息
|
|
|
+ callUserData(iMsg.getFrameCnt());
|
|
|
+ return;
|
|
|
}
|
|
|
|
|
|
+ if (0x07 == response.getCOT()) {
|
|
|
+ log.info("文件内容:" + iMsg.getDataContentString());
|
|
|
+
|
|
|
+ //解析光功率预测文件
|
|
|
+ ForcastModel forcastModel = forcastService.readForcastFileString(iMsg.getDataContentString());
|
|
|
+ if (forcastModel != null) {
|
|
|
+// log.info(forcastModel.dataType + ", size = " + forcastModel.getForcastData().size());
|
|
|
+ forcastService.putForcastModel(forcastModel);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (iMsg.getFrameCnt() > 1) {
|
|
|
+ //todo: 发送文件传输结束确认消息
|
|
|
+ Iec102Message fileEndMsg = BasicInstruction102.createFileEndMessage(iMsg.getFrameCnt(), iMsg.getDataContent().length);
|
|
|
+ sendMessage(fileEndMsg);
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
}
|
|
|
} catch (Exception ex) {
|
|
|
log.error(ex.getMessage());
|
|
@@ -148,6 +180,52 @@ public class Iec102Client {
|
|
|
}
|
|
|
|
|
|
|
|
|
+ private Iec102Message checkFileMessage(Iec102Message newMsg) throws UnsupportedEncodingException {
|
|
|
+ String mapKey = newMsg.getDataFile();
|
|
|
+ if (msgMap == null) {
|
|
|
+ msgMap = new HashMap<>();
|
|
|
+ }
|
|
|
+ if (msgMap.containsKey(mapKey)) {
|
|
|
+ Iec102Message oldMsg = msgMap.get(mapKey);
|
|
|
+ oldMsg = mergeFileMessage(oldMsg, newMsg);
|
|
|
+ return oldMsg;
|
|
|
+ }else {
|
|
|
+ if (newMsg.getCOT() == 0x08) {
|
|
|
+ newMsg.setFrameCnt(2); //召唤消息在文件消息之前
|
|
|
+ //不完整的新文件
|
|
|
+ msgMap.put(newMsg.getDataFile(), newMsg);
|
|
|
+ }
|
|
|
+
|
|
|
+ return newMsg;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ private Iec102Message mergeFileMessage(Iec102Message oldMsg, Iec102Message newMsg) throws UnsupportedEncodingException {
|
|
|
+ String newMsgData = newMsg.getDataContentString();
|
|
|
+ byte[] newBytes;
|
|
|
+ if (newMsgData.startsWith("#")) {
|
|
|
+ newBytes = ByteUtil.byteMergerWith0D0A(oldMsg.getDataContent(), newMsg.getDataContent());
|
|
|
+ } else {
|
|
|
+ newBytes = ByteUtil.byteMerger(oldMsg.getDataContent(), newMsg.getDataContent());
|
|
|
+ }
|
|
|
+
|
|
|
+ oldMsg.setDataContent(newBytes);
|
|
|
+ oldMsg.setCOT(newMsg.getCOT());
|
|
|
+ oldMsg.setFrameCnt(oldMsg.getFrameCnt() + 1);
|
|
|
+
|
|
|
+ return oldMsg;
|
|
|
+ }
|
|
|
+
|
|
|
+//
|
|
|
+// private void mergeFileMessage2(Iec102Message iec102Message) {
|
|
|
+// if (longFileMessage == null)
|
|
|
+// longFileMessage = iec102Message;
|
|
|
+// else {
|
|
|
+// byte[] newBytes = ByteUtil.byteMergerWith0D0A(longFileMessage.getDataContent(), iec102Message.getDataContent());
|
|
|
+// longFileMessage.setDataContent(newBytes);
|
|
|
+// }
|
|
|
+// }
|
|
|
+
|
|
|
private void sleep(int millionSencond) {
|
|
|
try {
|
|
|
Thread.sleep(millionSencond);
|