From 046b45b3f45430101fdfb9ee71ab8510ab5a8010 Mon Sep 17 00:00:00 2001
From: xuhy <3313886187@qq.com>
Date: 星期四, 10 四月 2025 17:17:15 +0800
Subject: [PATCH] 监管平台

---
 ruoyi-service/ruoyi-integration/src/main/java/com/ruoyi/integration/drainage/TCECSuperviseUtil.java |  264 +++++++++++++++++++++++++++++++++++-----------------
 1 files changed, 178 insertions(+), 86 deletions(-)

diff --git a/ruoyi-service/ruoyi-integration/src/main/java/com/ruoyi/integration/drainage/TCECSuperviseUtil.java b/ruoyi-service/ruoyi-integration/src/main/java/com/ruoyi/integration/drainage/TCECSuperviseUtil.java
index fb87f7c..0ed8892 100644
--- a/ruoyi-service/ruoyi-integration/src/main/java/com/ruoyi/integration/drainage/TCECSuperviseUtil.java
+++ b/ruoyi-service/ruoyi-integration/src/main/java/com/ruoyi/integration/drainage/TCECSuperviseUtil.java
@@ -6,16 +6,27 @@
 import com.alibaba.fastjson.JSON;
 import com.alibaba.fastjson.JSONObject;
 import com.alibaba.nacos.common.utils.JacksonUtils;
-import com.ruoyi.common.core.utils.SpringUtils;
+import com.fasterxml.jackson.core.JsonProcessingException;
+import com.fasterxml.jackson.databind.MapperFeature;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.ruoyi.common.core.utils.HttpUtils;
+import com.ruoyi.common.redis.service.RedisService;
 import com.ruoyi.integration.drainage.model.*;
-import com.ruoyi.integration.drainage.model.enu.InterfaceUrlEnum;
+import com.ruoyi.integration.drainage.util.AesEncryption;
+import com.ruoyi.integration.drainage.util.HMacMD5Util;
+import com.ruoyi.integration.drainage.util.SequenceGenerator;
 import com.ruoyi.other.api.domain.Operator;
-import com.ruoyi.other.api.feignClient.OperatorClient;
 import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Component;
+import org.springframework.util.StringUtils;
 
 import java.time.LocalDateTime;
-import java.time.ZoneOffset;
 import java.time.format.DateTimeFormatter;
+import java.util.List;
+import java.util.concurrent.TimeUnit;
+
+import static com.ruoyi.integration.drainage.TCECUtil.getToken;
 
 /**
  * 中电联TCEC标准
@@ -23,10 +34,29 @@
  * @Date 2025/1/21 11:48
  */
 @Slf4j
+@Component
 public class TCECSuperviseUtil {
-	
-	private static OperatorClient operatorClient = SpringUtils.getBean(OperatorClient.class);
+	// 测试环境
+	private static final String OperatorID = "MA01H3BQ2";
+	private static final String OperatorSecret = "f1331ef0b37c2d1b";
+	private static final String SigSecret = "a6fedf0e1b27d6f7";
+	private static final String DataSecret = "50a61b93919c9604";
+	private static final String DataSecretIV = "7c8ac6861661d584";
+
+
+	private static final String OurDataSecret = "50a61b93919c9605";
+	private static final String OurDataSecretIV = "7c8ac6861661d585";
+	private static final String OurSigSecret = "a6fedf0e1b27d6f6";
+	private static final String OurOperatorID = "MA01H3BQ3";
+	private static final String OurOperatorSecret = "f1331ef0b37c2d1a";
+	@Autowired
+	private RedisService redisService;
+	public static final String TOKEN_KEY = "charge_token:";
 	private final static String url = "https://dev-gov-hlht-sc.unievbj.com/evcs/v1.0.0/";
+	/**
+	 * 获取token
+	 */
+	private final static String query_token = "/query_token";
 	/**
 	 * 推送充电设备接口状态信息
 	 */
@@ -39,77 +69,53 @@
 	 *推送充电订单信息
 	 */
 	private final static String supervise_notification_charge_order_info = "/supervise_notification_charge_order_info";
+	private final static String supervise_notification_operation_stats_info = "/supervise_notification_operation_stats_info";
+	private final static String supervise_notification_realtime_power_info = "/supervise_notification_realtime_power_info";
+	private final static String supervise_notification_station_info = "/supervise_notification_station_info";
 
 
-	
+
 	/**
 	 * 获取token
 	 */
-	public static String queryToken(Operator operator){
-		HttpRequest post = HttpUtil.createPost(operator.getUrl() + InterfaceUrlEnum.QUERY_TOKEN.getUrl());
-		JSONObject info = new JSONObject();
-		info.put("OperatorID", operator.getOurOperatorId());
-		info.put("OperatorSecret", operator.getOperatorSecret());
-		Long timeStamp = Long.valueOf(LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyyMMddHHmmss")));
-		post.contentType("application/json;charset=utf-8");
-		BaseRequest baseRequest = new BaseRequest();
-		baseRequest.setOperatorID(operator.getOurOperatorId());
-		baseRequest.setTimeStamp(timeStamp);
-		baseRequest.setSeq("0001");
-		String jsonString = JacksonUtils.toJson(info);
-		String encrypt = AESUtil.encrypt(jsonString, operator.getDataSecret(), operator.getDataSecretIv());
-		baseRequest.setData(encrypt);
-		baseRequest.setOperator(operator);
-//		baseRequest.setSig(buildSign(baseRequest));
-		String request_json = JacksonUtils.toJson(baseRequest);
-		log.info("获取三方平台授权token请求地址:" + post.getUrl());
-		log.info("获取三方平台授权token请求参数:" + request_json);
-		log.info("获取三方平台授权token请求Data:" + jsonString);
-		post.body(request_json);
-		HttpResponse execute = post.execute();
-		if(200 != execute.getStatus()){
-			log.error("获取三方平台授权token失败:" + execute.body());
-			return null;
+	public String queryToken(){
+		String token = redisService.getCacheObject(TOKEN_KEY);
+		if(StringUtils.hasLength(token)){
+			return token;
 		}
-		log.info("获取三方平台授权token响应参数:" + execute.body());
-		BaseResult baseResult = JSON.parseObject(execute.body(), BaseResult.class);
-		Integer Ret = baseResult.getRet();
-		if(0 != Ret){
-			log.error("获取三方平台授权token失败:" + baseResult.getMsg());
-			return null;
-		}
-		//解密参数
-		String decrypt = AESUtil.decrypt(baseResult.getData(), operator.getDataSecret(), operator.getDataSecretIv());
-		log.info("获取三方平台授权token响应Data:" + decrypt);
-		QueryTokenResult queryTokenResult = JSON.parseObject(decrypt, QueryTokenResult.class);
-		String token = queryTokenResult.getAccessToken();
-		Long tokenAvailableTime = LocalDateTime.now().toEpochSecond(ZoneOffset.UTC) + queryTokenResult.getTokenAvailableTime();
-		operator.setAccessToken(token);
-		operator.setTokenAvailableTime(tokenAvailableTime);
-		operatorClient.editOperator(operator);
+		JSONObject jsonObject = new JSONObject();
+		jsonObject.put("OperatorID", OperatorID);
+		jsonObject.put("OperatorSecret", OperatorSecret);
+		String params = jsonObject.toJSONString();
+		// 参数加密
+		String data = AesEncryption.encrypt(DataSecret, DataSecretIV, params);
+		// 获取签名
+		String timeStamp = System.currentTimeMillis() + "";
+		SequenceGenerator generator = new SequenceGenerator();
+		String nextSequence = generator.getNextSequence();
+		String hmacMD5 = HMacMD5Util.getHMacMD5(OperatorID,timeStamp, data,nextSequence,SigSecret);
+		jsonObject = new JSONObject();
+		jsonObject.put("OperatorID", OperatorID);
+		jsonObject.put("Data", data);
+		jsonObject.put("TimeStamp", timeStamp);
+		jsonObject.put("Seq", nextSequence);
+		jsonObject.put("Sig", hmacMD5);
+		String result = HttpUtils.sendPost(url+query_token, jsonObject.toJSONString());
+		String string = JSONObject.parseObject(result).getString("Data");
+		String decrypt = AesEncryption.decrypt(DataSecret, DataSecretIV, string);
+		JSONObject tokenResult = JSONObject.parseObject(decrypt);
+		token = tokenResult.getString("AccessToken");
+		Integer TokenAvailableTime = tokenResult.getInteger("TokenAvailableTime");
+		redisService.setCacheObject(TOKEN_KEY,token, (long) (TokenAvailableTime - 60), TimeUnit.SECONDS);
 		return token;
 	}
-	
-	
-	/**
-	 * 获取token
-	 * @return
-	 */
-	public static String getToken(Operator operator){
-		if(null != operator.getTokenAvailableTime() && operator.getTokenAvailableTime() > LocalDateTime.now().toEpochSecond(ZoneOffset.UTC)){
-			return operator.getAccessToken();
-		}else{
-			return queryToken(operator);
-		}
-	}
-	
-	
-	
+
+
 	/**
 	 * 设备状态变化推送
 	 * @param info
 	 */
-	public static NotificationStationStatusResult notificationStationStatus(Operator operator, ConnectorStatusInfo info) {
+	public NotificationStationStatusResult notificationStationStatus(Operator operator, ConnectorStatusInfo info) {
 		HttpRequest post = HttpUtil.createPost(url+supervise_notification_station_status);
 		buildBody(post, info, operator);
 		HttpResponse execute = post.execute();
@@ -125,19 +131,19 @@
 			return null;
 		}
 		//解密参数
-		String decrypt = AESUtil.decrypt(baseResult.getData(), operator.getDataSecret(), operator.getDataSecretIv());
+		String decrypt = AESUtil.decrypt(baseResult.getData(), DataSecret, DataSecretIV);
 		log.info("推送三方平台设备状态Data:" + decrypt);
 		NotificationStationStatusResult notificationStationStatusResult = JSON.parseObject(decrypt, NotificationStationStatusResult.class);
 		return notificationStationStatusResult;
 	}
-	
-	
+
+
 	/**
 	 * 推动充电状态
 	 * @param info
 	 * @return
 	 */
-	public static NotificationEquipChargeStatusResult notificationSupEquipChargeStatus(Operator operator, SupEquipChargeStatus info){
+	public NotificationEquipChargeStatusResult notificationSupEquipChargeStatus(Operator operator, SupEquipChargeStatus info){
 		HttpRequest post = HttpUtil.createPost(url+supervise_notification_equip_charge_status);
 		buildBody(post, info, operator);
 		HttpResponse execute = post.execute();
@@ -153,19 +159,19 @@
 			return null;
 		}
 		//解密参数
-		String decrypt = AESUtil.decrypt(baseResult.getData(), operator.getDataSecret(), operator.getDataSecretIv());
+		String decrypt = AESUtil.decrypt(baseResult.getData(), DataSecret, DataSecretIV);
 		log.info("推送三方平台充电状态Data:" + decrypt);
 		NotificationEquipChargeStatusResult notificationEquipChargeStatusResult = JSON.parseObject(decrypt, NotificationEquipChargeStatusResult.class);
 		return notificationEquipChargeStatusResult;
 	}
-	
-	
+
+
 	/**
 	 * 推送充电订单信息
 	 * @param info
 	 * @return
 	 */
-	public static NotificationChargeOrderInfoResult notificationChargeOrderInfo(Operator operator, SupChargeOrderInfo info){
+	public NotificationChargeOrderInfoResult notificationChargeOrderInfo(Operator operator, SupChargeOrderInfo info){
 		HttpRequest post = HttpUtil.createPost(url+supervise_notification_charge_order_info);
 		buildBody(post, info, operator);
 		HttpResponse execute = post.execute();
@@ -181,31 +187,42 @@
 			return null;
 		}
 		//解密参数
-		String decrypt = AESUtil.decrypt(baseResult.getData(), operator.getDataSecret(), operator.getDataSecretIv());
+		String decrypt = AESUtil.decrypt(baseResult.getData(), DataSecret, DataSecretIV);
 		log.info("推送三方平台充电订单信息Data:" + decrypt);
 		NotificationChargeOrderInfoResult notificationChargeOrderInfoResult = JSON.parseObject(decrypt, NotificationChargeOrderInfoResult.class);
 		return notificationChargeOrderInfoResult;
 	}
-	
-	
+
+
 	/**
 	 * 构建请求参数和消息头
 	 * @param post
 	 * @param o
 	 */
-	public static void buildBody(HttpRequest post, Object o, Operator operator){
+	public void buildBody(HttpRequest post, Object o, Operator operator){
 		Long timeStamp = Long.valueOf(LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyyMMddHHmmss")));
 		post.contentType("application/json;charset=utf-8");
-		post.header("Authorization", "Bearer " + getToken(operator));
+		post.header("Authorization", "Bearer " + queryToken());
 		BaseRequest baseRequest = new BaseRequest();
 		baseRequest.setOperatorID(operator.getOurOperatorId());
 		baseRequest.setTimeStamp(timeStamp);
 		baseRequest.setSeq("0001");
-		String jsonString = JacksonUtils.toJson(o);
-		String encrypt = AESUtil.encrypt(jsonString, operator.getDataSecret(), operator.getDataSecretIv());
+		// 使用 Jackson 转换为 JSON
+		ObjectMapper objectMapper = new ObjectMapper();
+		objectMapper.enable(MapperFeature.USE_ANNOTATIONS); // 启用注解支持
+        String jsonString = null;
+        try {
+			jsonString = objectMapper.writeValueAsString(o);
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+		String encrypt = AesEncryption.encrypt(jsonString, DataSecret, DataSecretIV);
 		baseRequest.setData(encrypt);
 		baseRequest.setOperator(operator);
-//		baseRequest.setSig(buildSign(baseRequest));
+		SequenceGenerator generator = new SequenceGenerator();
+		String nextSequence = generator.getNextSequence();
+		String hmacMD5 = HMacMD5Util.getHMacMD5(OperatorID,timeStamp+"", encrypt,nextSequence,SigSecret);
+		baseRequest.setSig(hmacMD5);
 		String request_json = JacksonUtils.toJson(baseRequest);
 		post.body(request_json);
 		log.info("推送三方平台请求地址:" + post.getUrl());
@@ -213,11 +230,86 @@
 		log.info("推送三方平台请求Data:" + jsonString);
 	}
 
-	
-	
+	/**
+	 * 推送充电用能统计
+	 * @param info
+	 * @return
+	 */
+	public void superviseNotificationOperationStatsInfo(StationStatsInfoResult info){
+		Operator operator = new Operator();
+		HttpRequest post = HttpUtil.createPost(url+supervise_notification_operation_stats_info);
+		buildBody(post, info, operator);
+		HttpResponse execute = post.execute();
+		if(200 != execute.getStatus()){
+			log.error("推送充电用能统计失败:" + execute.body());
+			return ;
+		}
+		log.info("推送充电用能统计信息响应:" + execute.body());
+		BaseResult baseResult = JSON.parseObject(execute.body(), BaseResult.class);
+		Integer Ret = baseResult.getRet();
+		if(0 != Ret){
+			log.error("推送充电用能统计信息失败:" + baseResult.getMsg());
+			return ;
+		}
+		//解密参数
+		String decrypt = AESUtil.decrypt(baseResult.getData(), DataSecret, DataSecretIV);
+		log.info("推送充电用能统计信息Data:" + decrypt);
+
+	}
+	/**
+	 * 推送充换电站信息
+	 * @param info
+	 * @return
+	 */
+	public void superviseNotificationStationInfo(SupStationInfoResult info){
+		Operator operator = new Operator();
+		HttpRequest post = HttpUtil.createPost(url+supervise_notification_station_info);
+		buildBody(post, info, operator);
+		HttpResponse execute = post.execute();
+		if(200 != execute.getStatus()){
+			log.error("推送充换电站信息失败:" + execute.body());
+			return ;
+		}
+		log.info("推送充换电站信息信息响应:" + execute.body());
+		BaseResult baseResult = JSON.parseObject(execute.body(), BaseResult.class);
+		Integer Ret = baseResult.getRet();
+		if(0 != Ret){
+			log.error("推送充换电站信息失败:" + baseResult.getMsg());
+			return ;
+		}
+		//解密参数
+		String decrypt = AESUtil.decrypt(baseResult.getData(), DataSecret, DataSecretIV);
+		log.info("推送充换电站信息信息Data:" + decrypt);
+	}
+	/**
+	 * 充换电站功率信息
+	 * @param info
+	 * @return
+	 */
+	public void superviseNotificationRealtimePowerInfo(SupStationPowerInfoResult info){
+		Operator operator = new Operator();
+		HttpRequest post = HttpUtil.createPost(url+supervise_notification_realtime_power_info);
+		buildBody(post, info, operator);
+		HttpResponse execute = post.execute();
+		if(200 != execute.getStatus()){
+			log.error("推送充换电站实时功率失败:" + execute.body());
+			return ;
+		}
+		log.info("推送充换电站实时功率信息响应:" + execute.body());
+		BaseResult baseResult = JSON.parseObject(execute.body(), BaseResult.class);
+		Integer Ret = baseResult.getRet();
+		if(0 != Ret){
+			log.error("推送充换电站实时功率信息失败:" + baseResult.getMsg());
+			return ;
+		}
+		//解密参数
+		String decrypt = AESUtil.decrypt(baseResult.getData(), DataSecret, DataSecretIV);
+		log.info("推送充换电站实时功率信息Data:" + decrypt);
+	}
+
 	public static void main(String[] args) {
 		BaseModel model = new BaseModel();
-		model.setOperatorID("MA25CNM38");
+		model.setOperatorID("MA01H3BQ2");
 		model.setData("AoArdDDcmHcmOMkCLHodTpY1xLtt9yhLqxvKPyfdlmEOBj1LJnQM+Z4JOZllt3Pj9rubfgxJ51zMAfzquQegJzHGAT9Y7JrKFzFe6jGtXo0=");
 		model.setTimeStamp(20250205120800L);
 		model.setSeq("0001");
@@ -232,8 +324,8 @@
 		// 打印计算得到的签名Sig
 		String s = SignUtil.bytesToHexString(hmacMd5);
 		System.err.println(s);
-		
-		
+
+
 //		BaseModel model = new BaseModel();
 //		model.setOperatorID("MA25CNM38");
 //		model.setData("xQYYEPiwoc4JENnQsF50qP6Tmnj7uU/AshWn3QbIgrMIk8zvFpy6fWLIiLzQHcLPFCihqH1uzoNhGVNcS8Wvf5gIx9+cWJvYtjwtsjAGMEwNQ6y+rF1jjKnqCNcQbHbhqJ2l76sD54QCFKyenKxymrXri19bDV8hizn3nBw+jcHBtqKHWzjdSHwPDWPIw4rXnrbWLya2PXVgYtVg93WPtliw+0xsqW0BxJamE70/Ilbg4wB5fOAOvXmbLFI+v9jRjpcA7ImhprqPMYkXYAfa2QlICrJEBK7DfpJvRMzKzeT90kOAITqIqFvbfV0ti+greRqToLP6ni4o6w3ID42UmKasIpLTnx+qtR4bVTBTRuDzXvNXDA92RqHaJO21xbSen1eHN/yGfrchBIM0gmav7EpaYAerfU4ubnmJ6xujR6okYepJwBD5DiSIrObfS+GWVPRekvnjvm7BC02NXTrGKA==");

--
Gitblit v1.7.1