From 41e05da9a811cd9394ce9bddf5edcb7cde58db06 Mon Sep 17 00:00:00 2001
From: 无关风月 <443237572@qq.com>
Date: 星期四, 06 三月 2025 16:49:36 +0800
Subject: [PATCH] Merge branch 'master' of https://gitee.com/xiaochen991015/xizang

---
 ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/TencentCosUtil.java              |  215 +++--
 ruoyi-admin/src/main/resources/application-test.yml                                      |   17 
 ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/BankOutController.java            |  190 +++++
 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TFlowManagementServiceImpl.java |    2 
 ruoyi-system/src/main/resources/mapper/system/TBillMapper.xml                            |   10 
 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysFileServiceImpl.java         |   42 +
 ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/PdfUtils.java                    |    2 
 bankapi/src/main/java/com/taxi591/bankapi/dto/QueryBillResponse.java                     | 1027 +++++++++++++++++++++++++++
 ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysFileMapper.java                    |   16 
 ruoyi-system/src/main/resources/mapper/system/TFlowManagementMapper.xml                  |    8 
 ruoyi-system/src/main/java/com/ruoyi/system/model/TFile.java                             |   81 ++
 ruoyi-common/src/main/java/com/ruoyi/common/config/FileUploadConfig.java                 |    2 
 ruoyi-system/src/main/java/com/ruoyi/system/service/SysFileService.java                  |   17 
 ruoyi-applet/src/main/java/com/ruoyi/web/controller/api/TInformationController.java      |    4 
 ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/COSController.java                |  155 +++
 ruoyi-admin/pom.xml                                                                      |   10 
 ruoyi-admin/src/main/java/com/ruoyi/web/core/config/FileUploaderConfig.java              |   67 +
 bankapi/src/main/java/com/taxi591/bankapi/dto/QueryBillRequest.java                      |  317 ++++++++
 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java           |   12 
 ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/SysFileController.java            |   22 
 ruoyi-applet/src/main/resources/application-test.yml                                     |    2 
 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TPayOrderServiceImpl.java       |   13 
 ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TContractController.java          |    1 
 ruoyi-admin/src/main/resources/application-prod.yml                                      |    2 
 ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TContractServiceImpl.java       |    8 
 ruoyi-framework/src/main/java/com/ruoyi/framework/config/SecurityConfig.java             |    2 
 ruoyi-system/src/main/java/com/ruoyi/system/model/TContract.java                         |    2 
 ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/TokenService.java          |    8 
 ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/WordUtil.java                    |    2 
 ruoyi-system/src/main/resources/mapper/system/TBankFlowMapper.xml                        |    6 
 30 files changed, 2,136 insertions(+), 126 deletions(-)

diff --git a/bankapi/src/main/java/com/taxi591/bankapi/dto/QueryBillRequest.java b/bankapi/src/main/java/com/taxi591/bankapi/dto/QueryBillRequest.java
new file mode 100644
index 0000000..841bc19
--- /dev/null
+++ b/bankapi/src/main/java/com/taxi591/bankapi/dto/QueryBillRequest.java
@@ -0,0 +1,317 @@
+package com.taxi591.bankapi.dto;
+
+import com.alibaba.fastjson2.annotation.JSONField;
+import com.fasterxml.jackson.databind.annotation.JsonSerialize;
+
+import java.io.Serializable;
+
+/**
+ *  直连商户平台账单查询输入对象,需要转换成json串发送给第三方系统
+ *  @author DELL
+ * 
+ */
+@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+public class QueryBillRequest implements Serializable {
+	
+	private static final long serialVersionUID = 1L;
+	
+	/** 格式 */
+	@JSONField(name = "format")
+	private String format;
+	
+	/** 消息 */
+	@JSONField(name = "message")
+	private Message message;
+
+	@Override
+	public String toString() {
+		return "QueryBillRequest[format=" + format + ",message=" + message.toString() + "]";
+	}
+
+	public String getFormat() {
+		return format;
+	}
+
+	@JSONField(name = "format")
+	public void setFormat(String format) {
+		this.format = format;
+	}
+
+	public Message getMessage() {
+		return message;
+	}
+
+	@JSONField(name = "message")
+	public void setMessage(Message message) {
+		this.message = message;
+	}
+
+	/**
+	 *  
+	 * 账单查询内部消息对象实体message内部类
+	 * 
+	 */
+	@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+	public class Message implements Serializable {
+		
+		private static final long serialVersionUID = 1L;
+		
+		/** 消息头部 */
+		@JSONField(name = "head")
+		private Head head;
+		
+		/** 消息体  */
+		@JSONField(name = "info")
+		private Info info;
+
+		@Override
+		public String toString() {
+			return "QueryBillRequest.Message[head=" + head.toString() + ",info=" + info.toString() + "]";
+		}
+
+		public Head getHead() {
+			return head;
+		}
+
+		@JSONField(name = "head")
+		public void setHead(Head head) {
+			this.head = head;
+		}
+
+		public Info getInfo() {
+			return info;
+		}
+
+		@JSONField(name = "info")
+		public void setInfo(Info info) {
+			this.info = info;
+		}
+
+		/**
+		 *  message子对象head消息头内部类
+		 */
+		@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+		public class Head implements Serializable {
+			
+			private static final long serialVersionUID = 1L;
+			
+			/**  渠道编码 */
+			@JSONField(name = "channel")
+			private String channel;
+			
+			/**  交易码  */
+			@JSONField(name = "transCode")
+			private String transCode;
+			
+			/**  交易上行下送标志位  */
+			@JSONField(name = "transFlag")
+			private String transFlag;			
+			
+			/**  缴费中心交易序列号 */
+			@JSONField(name = "transSeqNum")
+			private String transSeqNum;
+			
+			/**   时间戳  */
+			@JSONField(name = "timeStamp")
+			private String timeStamp;
+			
+			/**   4为分行iGoal码  */
+			@JSONField(name = "branchCode")
+			private String branchCode;
+
+			@Override
+			public String toString() {
+				return "QueryBillRequest.Message.Head[channel=" + channel +",transCode=" + transCode + 
+						",transFlag=" + transFlag + ",transSeqNum=" + transSeqNum + ",timestamp=" + timeStamp + ",branchCode=" + branchCode + "]";
+			}
+
+			public String getChannel() {
+				return channel;
+			}
+
+			@JSONField(name = "channel")
+			public void setChannel(String channel) {
+				this.channel = channel;
+			}
+
+			public String getTransFlag() {
+				return transFlag;
+			}
+
+			@JSONField(name = "transFlag")
+			public void setTransFlag(String transFlag) {
+				this.transFlag = transFlag;
+			}
+			
+			public String getTransCode() {
+				return transCode;
+			}
+
+			@JSONField(name = "transCode")
+			public void setTransCode(String transCode) {
+				this.transCode = transCode;
+			}
+			
+			public String getTransSeqNum() {
+				return transSeqNum;
+			}
+
+			@JSONField(name = "transSeqNum")
+			public void setTransSeqNum(String transSeqNum) {
+				this.transSeqNum = transSeqNum;
+			}
+
+
+			public String getTimeStamp() {
+				return timeStamp;
+			}
+
+			@JSONField(name = "timeStamp")
+			public void setTimeStamp(String timeStamp) {
+				this.timeStamp = timeStamp;
+			}
+			
+			public String getBranchCode() {
+				return branchCode;
+			}
+
+			@JSONField(name = "branchCode")
+			public void setBranchCode(String branchCode) {
+				this.branchCode = branchCode;
+			}
+		}
+
+		/**
+		 * message子对象info消息实体内部类
+		 */
+		@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+		public class Info implements Serializable {
+			
+			private static final long serialVersionUID = 1L;
+
+			/** 缴费项目编号*/
+			@JSONField(name = "epayCode")
+			private String epayCode;
+			
+			/** 第三方商户编号*/
+			@JSONField(name = "merchantId")
+			private String merchantId;
+			
+			/** 输入要素1*/
+			@JSONField(name = "input1")
+			private String input1;
+			
+			/** 输入要素2*/
+			@JSONField(name = "input2")
+			private String input2;
+			
+			/** 输入要素3*/
+			@JSONField(name = "input3")
+			private String input3;
+			
+			/** 输入要素4*/
+			@JSONField(name = "input4")
+			private String input4;
+			
+			/** 输入要素5*/
+			@JSONField(name = "input5")
+			private String input5;
+			
+			/** 16位客户号*/
+			@JSONField(name = "userId")
+			private String userId;
+			
+			/** 缴费中心流水号*/
+			@JSONField(name = "traceNo")
+			private String traceNo;
+
+			@Override
+			public String toString() {
+				return "QueryBillRequest.Message.Info[epayCode=" + epayCode + ",merchantId=" + merchantId + 
+						",input1=" + input1 + ",input2=" + input2 + ",input3=" + input3 + ",input4=" + input4 + ",input5=" + input5 + 
+						",userId=" + userId + ",traceNo=" + traceNo + "]";
+			}
+
+			public String getEpayCode() {
+				return epayCode;
+			}
+
+			@JSONField(name = "epayCode")
+			public void setEpayCode(String epayCode) {
+				this.epayCode = epayCode;
+			}
+			
+			public String getMerchantId() {
+				return merchantId;
+			}
+
+			@JSONField(name = "merchantId")
+			public void setMerchantId(String merchantId) {
+				this.merchantId = merchantId;
+			}
+
+			public String getInput1() {
+				return input1;
+			}
+
+			@JSONField(name = "input1")
+			public void setInput1(String input1) {
+				this.input1 = input1;
+			}
+
+			public String getInput2() {
+				return input2;
+			}
+
+			@JSONField(name = "input2")
+			public void setInput2(String input2) {
+				this.input2 = input2;
+			}
+
+			public String getInput3() {
+				return input3;
+			}
+
+			@JSONField(name = "input3")
+			public void setInput3(String input3) {
+				this.input3 = input3;
+			}
+
+			public String getInput4() {
+				return input4;
+			}
+
+			@JSONField(name = "input4")
+			public void setInput4(String input4) {
+				this.input4 = input4;
+			}
+
+			public String getInput5() {
+				return input5;
+			}
+
+			@JSONField(name = "input5")
+			public void setInput5(String input5) {
+				this.input5 = input5;
+			}
+			
+			public String getUserId() {
+				return userId;
+			}
+
+			@JSONField(name = "userId")
+			public void setUserId(String userId) {
+				this.userId = userId;
+			}
+			
+			public String getTraceNo() {
+				return traceNo;
+			}
+
+			@JSONField(name = "traceNo")
+			public void setTraceNo(String traceNo) {
+				this.traceNo = traceNo;
+			}
+		}
+	}
+}
\ No newline at end of file
diff --git a/bankapi/src/main/java/com/taxi591/bankapi/dto/QueryBillResponse.java b/bankapi/src/main/java/com/taxi591/bankapi/dto/QueryBillResponse.java
new file mode 100644
index 0000000..b33c9d1
--- /dev/null
+++ b/bankapi/src/main/java/com/taxi591/bankapi/dto/QueryBillResponse.java
@@ -0,0 +1,1027 @@
+package com.taxi591.bankapi.dto;
+
+import com.alibaba.fastjson2.annotation.JSONField;
+import com.fasterxml.jackson.databind.annotation.JsonSerialize;
+
+import java.io.Serializable;
+import java.util.ArrayList;
+
+/**
+ *  直连商户平台账单查询返回对象
+ *  @author DELL
+ * 
+ */
+@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+public class QueryBillResponse implements Serializable {
+	
+	private static final long serialVersionUID = 1L;
+	
+	/** 格式 */
+	@JSONField(name = "format")
+	private String format;
+	
+	/** 消息体 */
+	@JSONField(name = "message")
+	private Message message;
+	
+	
+	public QueryBillResponse(){
+		
+	}
+	
+	/**
+	 * 构造函数,通过输入对象,构造返回对象数据信息
+	 * @param request
+	 */
+	public QueryBillResponse(QueryBillRequest request) {
+		this.setFormat(request.getFormat());
+		this.setMessage(new Message(request.getMessage()));
+	}
+	
+	@Override
+	public String toString() {
+		return "QueryBillResponse[format=" + format + ",message=" + message==null?"":message.toString() + "]";
+	}
+	
+	public String getFormat() {
+		return format;
+	}
+
+	@JSONField(name = "format")
+	public void setFormat(String format) {
+		this.format = format;
+	}
+
+	public Message getMessage() {
+		return message;
+	}
+
+	@JSONField(name = "message")
+	public void setMessage(Message message) {
+		this.message = message;
+	}
+
+	/**
+	 *  
+	 * 账单查询内部消息对象返回实体message内部类
+	 * 
+	 */
+	@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+	public class Message implements Serializable {
+		
+		private static final long serialVersionUID = 1L;
+		
+		/** 消息头部 */
+		@JSONField(name = "head")
+		private Head head;
+		
+		/** 消息实体  */
+		@JSONField(name = "info")
+		private Info info;
+		
+		public Message() {
+			this.head = new Head();
+			this.info = new Info();
+		}
+		
+		public Message(QueryBillRequest.Message requestMessage){
+			this.setHead(new Head(requestMessage.getHead()));
+			this.setInfo(new Info(requestMessage.getInfo()));
+		}
+		
+		@Override
+		public String toString() {
+			return "QueryBillResponse.Message[head=" + head.toString() + ",info=" + info.toString() + "]";
+		}
+		
+		public Head getHead() {
+			return head;
+		}
+
+		@JSONField(name = "head")
+		public void setHead(Head head) {
+			this.head = head;
+		}
+
+		public Info getInfo() {
+			return info;
+		}
+
+		@JSONField(name = "info")
+		public void setInfo(Info info) {
+			this.info = info;
+		}
+
+		/**
+		 *  
+		 * 账单查询内部消息对象返回实体Head内部类
+		 * 
+		 */
+		@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+		public class Head implements Serializable {
+			
+			private static final long serialVersionUID = 1L;
+			
+			/**  渠道 */
+			@JSONField(name = "channel")
+			private String channel;
+			
+			/**  交易码  */
+			@JSONField(name = "transCode")
+			private String transCode;
+			
+			/**  交易上行下送标志  */
+			@JSONField(name = "transFlag")
+			private String transFlag;
+			
+			/**  缴费中心交易序列号 */
+			@JSONField(name = "transSeqNum")
+			private String transSeqNum;
+			
+			/** 时间戳  */
+			@JSONField(name = "timeStamp")
+			private String timeStamp;
+			
+			/**  查询返回码 */
+			@JSONField(name = "returnCode")
+			private String returnCode ;
+			
+			/**  返回提示信息  */
+			@JSONField(name = "returnMessage")
+			private String returnMessage;
+			
+
+			public Head(QueryBillRequest.Message.Head reqMessHead) {
+				this.setChannel(reqMessHead.getChannel());
+				this.setTransSeqNum(reqMessHead.getTransSeqNum());
+				this.setTransCode(reqMessHead.getTransCode());
+				this.setReturnCode("");
+				this.setReturnMessage("");
+				this.setTimeStamp("");
+			}
+
+			
+			public Head() {
+				
+			}
+			@Override
+			public String toString() {
+				return "QueryBillResponse.Message.Head[channel=" + channel  + ",transCode=" + transCode + ",transSeqNum=" + transSeqNum
+						+ ",timeStamp=" + timeStamp + ",returnCode=" + returnCode + ",returnMessage=" + returnMessage  + "]";
+			}
+			public String getChannel() {
+				return channel;
+			}
+
+			public String getTransCode() {
+				return transCode;
+			}
+
+			@JSONField(name = "transCode")
+			public void setTransCode(String transCode) {
+				this.transCode = transCode;
+			}
+			
+			public String getTransFlag() {
+				return transFlag;
+			}
+
+			@JSONField(name = "transFlag")
+			public void setTransFlag(String transFlag) {
+				this.transFlag = transFlag;
+			}
+			
+			@JSONField(name = "channel")
+			public void setChannel(String channel) {
+				this.channel = channel;
+			}
+
+			public String getTransSeqNum() {
+				return transSeqNum;
+			}
+
+			@JSONField(name = "transSeqNum")
+			public void setTransSeqNum(String transSeqNum) {
+				this.transSeqNum = transSeqNum;
+			}
+
+			
+			public String getTimeStamp() {
+				return timeStamp;
+			}
+
+			@JSONField(name = "timestamp")
+			public void setTimeStamp(String timeStamp) {
+				this.timeStamp = timeStamp;
+			}
+			
+			public String getReturnCode() {
+				return returnCode;
+			}
+
+			@JSONField(name = "returnCode")
+			public void setReturnCode(String returnCode) {
+				this.returnCode = returnCode;
+			}
+
+			public String getReturnMessage() {
+				return returnMessage;
+			}
+
+			@JSONField(name = "returnMessage")
+			public void setReturnMessage(String returnMessage) {
+				this.returnMessage = returnMessage;
+			}
+
+
+		}
+
+		/**
+		 *  
+		 * 账单查询内部消息对象返回实体Info内部类
+		 * 
+		 */
+		@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+		public class Info implements Serializable {
+			
+			private static final long serialVersionUID = 1L;
+			
+			/** 缴费项目唯一标识号*/
+			@JSONField(name = "epayCode")
+			private String epayCode;
+			
+			/** 直连商户第三方商户号*/
+			@JSONField(name = "merchantId")
+			private String merchantId;
+			
+			/** 缴费中心流水号*/
+			@JSONField(name = "traceNo")
+			private String traceNo;
+			
+			/** 输入要素1*/
+			@JSONField(name = "input1")
+			private String input1;
+			
+			/** 输入要素2*/
+			@JSONField(name = "input2")
+			private String input2;
+			
+			/** 输入要素3*/
+			@JSONField(name = "input3")
+			private String input3;
+			
+			/** 输入要素4*/
+			@JSONField(name = "input4")
+			private String input4;
+			
+			/** 输入要素5*/
+			@JSONField(name = "input5")
+			private String input5;
+			
+			/** 户主名称*/
+			@JSONField(name = "custName")
+			private String custName;
+			
+			/** 户主地址*/
+			@JSONField(name = "custAddress")
+			private String custAddress;
+			
+			/** 账单关联客户手机号,部分项目可根据该字段,要求客户做短信验证码校验后才显示账单结果*/
+			@JSONField(name = "cellPhone")
+			private String cellPhone;
+			
+			/** 缓存域信息*/
+			@JSONField(name = "cacheMem")
+			private String cacheMem;
+			
+			/** 备注字段*/
+			@JSONField(name = "remark")
+			private String remark;
+			
+			/** 跳转商户地址*/
+			@JSONField(name = "callBackUrl")
+			private String callBackUrl;
+			
+			/** 跳转商户地址超链接提示*/
+			@JSONField(name = "callBackText")
+			private String callBackText;
+			
+			/** 缴费金额计算规则*/
+			@JSONField(name = "amtRule")
+			private String amtRule;
+			
+			/** 子账单数量*/
+			@JSONField(name = "totalBillCount")
+			private String totalBillCount;
+			/** 定制附言信息(有定制电子回单附言信息的,需添加自定义定制附言信息字段)*/
+			@JSONField(name = "merchantRemark")
+			private String merchantRemark;
+			
+			/** 账单信息体*/
+			@JSONField(name = "bills")
+			private ArrayList<Bill> bills;
+			
+			
+			public Info() {
+				
+			}
+			public Info(QueryBillRequest.Message.Info reqMessInfo) {
+			    this.setEpayCode(reqMessInfo.getEpayCode());
+			    this.setInput1(reqMessInfo.getInput1());
+			    this.setInput2(reqMessInfo.getInput2());
+			    this.setInput3(reqMessInfo.getInput3());
+			    this.setInput4(reqMessInfo.getInput4());
+			    this.setInput5(reqMessInfo.getInput5());
+			    this.bills = new ArrayList<Bill>();
+			}
+			
+			/**
+			 * @return the merchantRemark
+			 */
+			public String getMerchantRemark() {
+				return merchantRemark;
+			}
+
+			/**
+			 * @param merchantRemark the merchantRemark to set
+			 */
+			@JSONField(name = "merchantRemark")
+			public void setMerchantRemark(String merchantRemark) {
+				this.merchantRemark = merchantRemark;
+			}
+
+			@Override
+			public String toString() {
+				return "QueryBillResponse.Message.Info[epayCode=" + epayCode + ", input1="
+						+ input1 + ", input2=" + input2 + ", input3=" + input3
+						+ ", input4=" + input4 + ", input5=" + input5
+						+ ", totalBillCount=" + totalBillCount + ",merchantRemark=" 
+						+ merchantRemark + ", bills=" + bills.toString() + "]";
+			}
+			
+			public String getEpayCode() {
+				return epayCode;
+			}
+
+			@JSONField(name = "epayCode")
+			public void setEpayCode(String epayCode) {
+				this.epayCode = epayCode;
+			}
+			
+			public String getMerchantId() {
+				return merchantId;
+			}
+
+			@JSONField(name = "merchantId")
+			public void setMerchantId(String merchantId) {
+				this.merchantId = merchantId;
+			}
+			
+			public String getTraceNo() {
+				return traceNo;
+			}
+
+			@JSONField(name = "traceNo")
+			public void setTraceNo(String traceNo) {
+				this.traceNo = traceNo;
+			}
+
+			public String getInput1() {
+				return input1;
+			}
+
+			@JSONField(name = "input1")
+			public void setInput1(String input1) {
+				this.input1 = input1;
+			}
+
+			public String getInput2() {
+				return input2;
+			}
+
+			@JSONField(name = "input2")
+			public void setInput2(String input2) {
+				this.input2 = input2;
+			}
+
+			public String getInput3() {
+				return input3;
+			}
+
+			@JSONField(name = "input3")
+			public void setInput3(String input3) {
+				this.input3 = input3;
+			}
+
+			public String getInput4() {
+				return input4;
+			}
+
+			@JSONField(name = "input4")
+			public void setInput4(String input4) {
+				this.input4 = input4;
+			}
+
+			public String getInput5() {
+				return input5;
+			}
+
+			@JSONField(name = "input5")
+			public void setInput5(String input5) {
+				this.input5 = input5;
+			}
+
+			public String getCustName() {
+				return custName;
+			}
+
+			@JSONField(name = "custName")
+			public void setCustName(String custName) {
+				this.custName = custName;
+			}
+			
+			public String getCustAddress() {
+				return custAddress;
+			}
+
+			@JSONField(name = "custAddress")
+			public void setCustAddress(String custAddress) {
+				this.custAddress = custAddress;
+			}
+			
+			public String getCellPhone() {
+				return cellPhone;
+			}
+			
+			@JSONField(name = "cellPhone")
+			public void setCellPhone(String cellPhone) {
+				this.cellPhone = cellPhone;
+			}
+			
+			public String getCacheMem() {
+				return cacheMem;
+			}
+
+			@JSONField(name = "cacheMem")
+			public void setCacheMem(String cacheMem) {
+				this.cacheMem = cacheMem;
+			}
+			
+			public String getRemark() {
+				return remark;
+			}
+
+			@JSONField(name = "remark")
+			public void setRemark(String remark) {
+				this.remark = remark;
+			}
+			
+			public String getCallBackUrl() {
+				return callBackUrl;
+			}
+
+			@JSONField(name = "callBackUrl")
+			public void setCallBackUrl(String callBackUrl) {
+				this.callBackUrl = callBackUrl;
+			}
+
+			public String getCallBackText() {
+				return callBackText;
+			}
+
+			@JSONField(name = "callBackText")
+			public void setCallBackText(String callBackText) {
+				this.callBackText = callBackText;
+			}
+
+			public String getAmtRule() {
+				return amtRule;
+			}
+
+			@JSONField(name = "amtRule")
+			public void setAmtRule(String amtRule) {
+				this.amtRule = amtRule;
+			}
+			
+			public String getTotalBillCount() {
+				return totalBillCount;
+			}
+
+			@JSONField(name = "totalBillCount")
+			public void setTotalBillCount(String totalBillCount) {
+				this.totalBillCount = totalBillCount;
+			}
+
+			public ArrayList<Bill> getBills() {
+				return bills;
+			}
+
+			/**
+			 *  @param bills
+			 *  设置账单循环域子账单
+			 */
+			@JSONField(name = "bills")
+			public void setBill(ArrayList<Bill> bills) {
+				this.bills = bills;
+			}
+
+
+			/**
+			 *  
+			 * 账单查询内部消息对象返回实体Bill内部类
+			 * 
+			 */
+			@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+			public class Bill implements Serializable {
+				
+				private static final long serialVersionUID = 1L;
+
+				/** 账单编号*/
+				@JSONField(name = "billNo")
+				private String billNo;
+				
+				/** 账单名称*/
+				@JSONField(name = "billName")
+				private String billName;
+				
+				/** 欠费金额*/
+				@JSONField(name = "oweAmt")
+				private String oweAmt;
+				
+				/** 手续费*/
+				@JSONField(name = "feeAmt")
+				private String feeAmt;
+				
+				/** 最小金额*/
+				@JSONField(name = "minAmt")
+				private String minAmt;
+				
+				/** 最大金额*/
+				@JSONField(name = "maxAmt")
+				private String maxAmt;
+				
+				/** 余额*/
+				@JSONField(name = "balance")
+				private String balance;
+				
+				/** 缴费账单到期日*/
+				@JSONField(name = "expireDate")
+				private String expireDate;
+				
+				/** 收款商户号*/
+				@JSONField(name = "rcvMerchantId")
+				private String rcvMerchantId;
+				
+				/** 收款账号*/
+				@JSONField(name = "rcvAcc")
+				private String rcvAcc;
+				
+				/** 分账模板号*/
+				@JSONField(name = "tempSplitAcc")
+				private String tempSplitAcc;		
+
+				/** 均匀时段缴费 */
+				@JSONField(name = "unitDetail")
+				private UnitDetail unitDetail;
+				
+				/** 选择套餐*/
+				@JSONField(name = "optionDetails")
+				private ArrayList<OptionDetail> optionDetails;
+				
+				/** 账单详情描述*/
+				@JSONField(name = "descDetails")
+				private ArrayList<DescDetail> descDetails;
+				/** 平台商户分账子商户循环域 */
+				@JSONField(name = "splitSubMerInfos")
+				private ArrayList<SplitSubMerInfo> splitSubMerInfos;
+				
+				/** 子账单必缴标志,多账单合并支付时默认选中不可不选  true:必选    false:非必选 */
+				@JSONField(name = "mustPayFlag")
+				private String mustPayFlag;
+				
+				
+				/**
+				 * @return
+				 */
+				public String getMustPayFlag() {
+					return mustPayFlag;
+				}
+				
+				/**
+				 * @param mustPayFlag
+				 */
+				@JSONField(name = "mustPayFlag")
+				public void setMustPayFlag(String mustPayFlag) {
+					this.mustPayFlag = mustPayFlag;
+				}
+				
+				/**
+				 * @return
+				 */
+				public String getTempSplitAcc() {
+					return tempSplitAcc;
+				}
+				/**
+				 * 获取平台商户分账子商户循环域信息
+				 * @return
+				 */
+				public ArrayList<SplitSubMerInfo> getSplitSubMerInfos() {
+					return splitSubMerInfos;
+				}
+				/**
+				 * 设置平台商户分账子商户循环域信息
+				 * @param splitSubMerInfos
+				 */
+				@JSONField(name = "splitSubMerInfos")
+				public void setSplitSubMerInfos(ArrayList<SplitSubMerInfo> splitSubMerInfos) {
+					this.splitSubMerInfos = splitSubMerInfos;
+				}
+				
+				public Bill() {
+					
+				}
+
+				@Override
+				public String toString() {
+					return "Bill [billNo=" + billNo + ", billName=" + billName
+							+ ", oweAmt=" + oweAmt + ", feeAmt=" + feeAmt
+							+ ", minAmt=" + minAmt + ", maxAmt=" + maxAmt
+							+ ", balance=" + balance + ", expireDate="
+							+ expireDate + ", rcvMerchantId=" + rcvMerchantId
+							+ ", rcvAcc=" + rcvAcc + ", tempSplitAcc="
+							+ tempSplitAcc + ", unitDetail=" + unitDetail
+							+ ", optionDetails=" + optionDetails
+							+ ", descDetails=" + descDetails
+							+ ", splitSubMerInfos=" + splitSubMerInfos
+							+ ", mustPayFlag=" + mustPayFlag + "]";
+				}
+				
+				public String getBillNo() {
+					return billNo;
+				}
+
+				@JSONField(name = "billNo")
+				public void setBillNo(String billNo) {
+					this.billNo = billNo;
+				}
+
+				public String getBillName() {
+					return billName;
+				}
+
+				@JSONField(name = "billName")
+				public void setBillName(String billName) {
+					this.billName = billName;
+				}
+
+				public String getOweAmt() {
+					return oweAmt;
+				}
+
+				@JSONField(name = "oweAmt")
+				public void setOweAmt(String oweAmt) {
+					this.oweAmt = oweAmt;
+				}
+
+				public String getFeeAmt() {
+					return feeAmt;
+				}
+
+				@JSONField(name = "feeAmt")
+				public void setFeeAmt(String feeAmt) {
+					this.feeAmt = feeAmt;
+				}
+
+				public String getMinAmt() {
+					return minAmt;
+				}
+
+				@JSONField(name = "minAmt")
+				public void setMinAmt(String minAmt) {
+					this.minAmt = minAmt;
+				}
+
+				public String getMaxAmt() {
+					return maxAmt;
+				}
+
+				@JSONField(name = "maxAmt")
+				public void setMaxAmt(String maxAmt) {
+					this.maxAmt = maxAmt;
+				}
+
+				public String getBalance() {
+					return balance;
+				}
+				@JSONField(name = "balance")
+				public void setBalance(String balance) {
+					this.balance = balance;
+				}
+
+				public String getExpireDate() {
+					return expireDate;
+				}
+				@JSONField(name = "expireDate")
+				public void setExpireDate(String expireDate) {
+					this.expireDate = expireDate;
+				}
+				
+				public String getRcvMerchantId() {
+					return rcvMerchantId;
+				}
+				@JSONField(name = "rcvMerchantId")
+				public void setRcvMerchantId(String rcvMerchantId) {
+					this.rcvMerchantId = rcvMerchantId;
+				}
+				
+				public String getRcvAcc() {
+					return rcvAcc;
+				}
+				@JSONField(name = "rcvAcc")
+				public void setRcvAcc(String rcvAcc) {
+					this.rcvAcc = rcvAcc;
+				}
+				
+				/**
+				 * 分账模板号
+				 * @return
+				 */
+				public String getTmpSplitAcc() {
+					return tempSplitAcc;
+				}
+				@JSONField(name = "tempSplitAcc")
+				public void setTempSplitAcc(String tempSplitAcc) {
+					this.tempSplitAcc = tempSplitAcc;
+				}
+				
+				public UnitDetail getUnitDetail() {
+					return unitDetail;
+				}
+				@JSONField(name = "unitDetail")
+				public void setUnitDetail(UnitDetail unitDetail) {
+					this.unitDetail = unitDetail;
+				}
+				
+				public ArrayList<DescDetail> getDescDetails() {
+					return descDetails;
+				}
+				/**
+				 *  设置账单描述详情键值对
+				 *  @param feeDetails
+				 */
+				@JSONField(name = "descDetails")
+				public void setDescDetails(ArrayList<DescDetail> descDetails) {
+					this.descDetails = descDetails;
+				}
+
+				public ArrayList<OptionDetail> getOptionDetails() {
+					return optionDetails;
+				}
+				@JSONField(name = "optionDetails")
+				public void setOptionDetails(ArrayList<OptionDetail> optionDetails) {
+					this.optionDetails = optionDetails;
+				}
+
+				/**
+				 * 均匀时段
+				 * 
+				 * @date 2017年12月21日
+				 */
+				@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+				public class UnitDetail implements Serializable {
+					
+					private static final long serialVersionUID = 1L;
+					
+					/** 单位名称 */
+					@JSONField(name = "unitName")
+					private String unitName;
+					
+					/** 单位金额 */
+					@JSONField(name = "unitAmount")
+					private String unitAmount;
+					
+					/** 最小单位数量 */
+					@JSONField(name = "minUnitNum")
+					private String minUnitNum;
+					public UnitDetail(){}
+					
+					public UnitDetail(String unitName, String unitAmount, String minUnitNum) {
+						super();
+						this.unitName = unitName;
+						this.unitAmount = unitAmount;
+						this.minUnitNum = minUnitNum;
+					}
+					public String getUnitName() {
+						return unitName;
+					}
+					
+					@JSONField(name = "unitAmount")
+					public void setUnitName(String unitName) {
+						this.unitName = unitName;
+					}
+					
+					public String getUnitAmount() {
+						return unitAmount;
+					}
+					
+					@JSONField(name = "unitAmount")
+					public void setUnitAmount(String unitAmount) {
+						this.unitAmount = unitAmount;
+					}
+					
+					public String getMinUnitNum() {
+						return minUnitNum;
+					}
+					
+					@JSONField(name = "minUnitNum")
+					public void setMinUnitNum(String minUnitNum) {
+						this.minUnitNum = minUnitNum;
+					}
+					
+					@Override
+					public String toString() {
+						return "QueryBillResponse.Message.Info.Bill.unitDetail[unitName=" + unitName
+								+ ", unitAmount=" + unitAmount + ", minUnitNum=" + minUnitNum + "]";
+					}
+				}
+
+				/**
+				 * @title 选择套餐循环
+				 *  
+				 *  @date 2017-12-18
+				 *  
+				 * 
+				 */
+				@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+				public class OptionDetail implements Serializable {
+					
+					private static final long serialVersionUID = 1L;
+					
+					/**套餐编号 根据编号排序*/
+					@JSONField(name = "optionCode")
+					private String optionCode;
+					
+					/**套餐名称*/
+					@JSONField(name = "optionName")
+					private String optionName;
+					
+					/**套餐金额*/
+					@JSONField(name = "optionAmt")
+					private String optionAmt;
+					
+					public OptionDetail(){
+						
+					}
+					
+					public OptionDetail(String optionCode, String optionName, String optionAmt) {
+						super();
+						this.optionCode = optionCode;
+						this.optionName = optionName;
+						this.optionAmt = optionAmt;
+					}
+
+					public String getOptionCode() {
+						return optionCode;
+					}
+					
+					@JSONField(name = "optionCode")
+					public void setOptionCode(String optionCode) {
+						this.optionCode = optionCode;
+					}
+					
+					public String getOptionName() {
+						return optionName;
+					}
+					
+					@JSONField(name = "optionName")
+					public void setOptionName(String optionName) {
+						this.optionName = optionName;
+					}
+
+					public String getOptionAmt() {
+						return optionAmt;
+					}
+					@JSONField(name = "optionAmt")
+					public void setOptionAmt(String optionAmt) {
+						this.optionAmt = optionAmt;
+					}
+
+					@Override
+					public String toString() {
+						return "QueryBillResponse.Message.Info.Bill.OptionDetail[optionCode=" + optionCode
+								+ ", optionName=" + optionName + ", optionAmt=" + optionAmt + "]";
+					}
+
+				}
+				/**
+				 * @title 账单详情
+				 *  
+				 *  @date 2017-12-18
+				 */
+				@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+				public class DescDetail implements Serializable {
+					
+					private static final long serialVersionUID = 1L;
+					
+					/**账单详情-名称*/
+					@JSONField(name = "sCpt")
+					private String sCpt;
+					
+					/**账单详情-取值*/
+					@JSONField(name = "sVal")
+					private String sVal;
+					
+					
+					public DescDetail(){
+						
+					}
+					
+					public DescDetail(String sCpt, String sVal) {
+						super();
+						this.sCpt = sCpt;
+						this.sVal = sVal;
+					}
+
+					public String getSCpt() {
+						return sCpt;
+					}
+
+					@JSONField(name = "sCpt")
+					public void setSCpt(String sCpt) {
+						this.sCpt = sCpt;
+					}
+
+					public String getSVal() {
+						return sVal;
+					}
+
+					@JSONField(name = "sVal")
+					public void setSVal(String sVal) {
+						this.sVal = sVal;
+					}
+
+					@Override
+					public String toString() {
+						return "QueryBillResponse.Message.Info.Bill.FeeDetail[sCpt=" + sCpt + ",sVal=" + sVal + "]";
+					}
+				}
+				/**
+				 * @title 平台商户分账子商户循环域
+				 *  
+				 * @date 2020-10-22
+				 *  
+				 * @author marui
+				 */
+				@JsonSerialize(include=JsonSerialize.Inclusion.NON_NULL)
+				public class SplitSubMerInfo implements Serializable {
+
+					private static final long serialVersionUID = 1L;
+
+					/**二级子商户号*/
+					@JSONField(name = "splitMerchantId")
+					private String splitMerchantId;
+
+					/**二级子商户分账金额*/
+					@JSONField(name = "splitAmount")
+					private String splitAmount;
+
+
+					public SplitSubMerInfo(){
+
+					}
+
+					public SplitSubMerInfo(String splitMerchantId, String splitAmount) {
+						super();
+						this.splitMerchantId = splitMerchantId;
+						this.splitAmount = splitAmount;
+					}
+
+					public String getSplitMerchantId() {
+						return splitMerchantId;
+					}
+
+					@JSONField(name = "splitMerchantId")
+					public void setSplitMerchantId(String splitMerchantId) {
+						this.splitMerchantId = splitMerchantId;
+					}
+
+					public String getSplitAmount() {
+						return splitAmount;
+					}
+
+					@JSONField(name = "splitAmount")
+					public void setSplitAmount(String splitAmount) {
+						this.splitAmount = splitAmount;
+					}
+
+
+					@Override
+					public String toString() {
+						return "QueryBillResponse.Message.Info.Bill.SplitSubMerInfo[splitMerchantId=" + splitMerchantId
+								+ ", splitAmount=" + splitAmount + "]";
+					}
+				}
+			}
+		}
+	}
+}
\ No newline at end of file
diff --git a/ruoyi-admin/pom.xml b/ruoyi-admin/pom.xml
index 5741488..96e7388 100644
--- a/ruoyi-admin/pom.xml
+++ b/ruoyi-admin/pom.xml
@@ -32,11 +32,11 @@
             <version>31.1-jre</version> <!-- 请根据需要选择合适的版本 -->
         </dependency>
         <!-- spring-boot-devtools -->
-<!--        <dependency>-->
-<!--            <groupId>org.springframework.boot</groupId>-->
-<!--            <artifactId>spring-boot-devtools</artifactId>-->
-<!--            <optional>true</optional> &lt;!&ndash; 表示依赖不会传递 &ndash;&gt;-->
-<!--        </dependency>-->
+        <dependency>
+            <groupId>org.springframework.boot</groupId>
+            <artifactId>spring-boot-devtools</artifactId>
+            <optional>true</optional> <!-- 表示依赖不会传递 -->
+        </dependency>
 
         <!-- swagger3-->
         <dependency>
diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/BankOutController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/BankOutController.java
index c474d4d..fd0274e 100644
--- a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/BankOutController.java
+++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/BankOutController.java
@@ -1,8 +1,23 @@
 package com.ruoyi.web.controller.api;
 
+import cn.hutool.core.date.DateUtil;
+import com.alibaba.fastjson.JSON;
+import com.alibaba.fastjson.TypeReference;
+import com.ruoyi.common.constant.AmountConstant;
+import com.ruoyi.common.utils.StringUtils;
+import com.ruoyi.system.dto.TBillDto;
+import com.ruoyi.system.model.TOrderBill;
+import com.ruoyi.system.model.TPayOrder;
 import com.ruoyi.system.service.TBillService;
+import com.ruoyi.system.service.TOrderBillService;
+import com.ruoyi.system.service.TPayOrderService;
 import com.taxi591.bankapi.dto.CovertPayBackResult;
+import com.taxi591.bankapi.dto.QueryBillRequest;
+import com.taxi591.bankapi.dto.QueryBillResponse;
 import com.taxi591.bankapi.service.BankService;
+import com.taxi591.bankapi.service.SignatureAndVerification;
+import lombok.extern.slf4j.Slf4j;
+import org.apache.commons.codec.binary.Base64;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.web.bind.annotation.PostMapping;
 import org.springframework.web.bind.annotation.RequestMapping;
@@ -10,9 +25,16 @@
 import org.springframework.web.bind.annotation.RestController;
 
 import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.math.BigDecimal;
+import java.math.RoundingMode;
+import java.util.*;
+import java.util.stream.Collectors;
+import java.util.stream.Stream;
 
 @RestController
 @RequestMapping("open/bank")
+@Slf4j
 public class BankOutController {
 
     @Autowired
@@ -20,6 +42,16 @@
 
     @Autowired
     TBillService tBillService;
+
+    @Autowired
+    SignatureAndVerification signatureAndVerification;
+
+    @Autowired
+    TOrderBillService orderBillService;
+
+    @Autowired
+    TPayOrderService payOrderService;
+
     @PostMapping(value = "payCallback")
     public @ResponseBody String payCallback(HttpServletRequest request){
         CovertPayBackResult result = bankService.covertPayCallBack(request, (billRequest) -> {
@@ -29,7 +61,165 @@
         return result.getBack();
     }
 
+    @PostMapping(value = "queryBill")
+    public void bills(HttpServletRequest request, HttpServletResponse httpServletResponse){
+        log.info("进入QueryBillController账单查询接口--------(金额规则为0的)-------");
 
+        String responseJson = null;
+        try {
+            //接收报文request返回截取并返回requestBody和使用base64解析后的requestBody
+            Map<String, String> requestMap = signatureAndVerification.requestBodyOfBase64(request);
+            //使用base64解析完成后的requestBody
+            String requestBodyOfDecoded = requestMap.get("requestBodyOfDecoded");
+            //解析前的requestBody
+            String requestBody = requestMap.get("requestBody");
+            //获取缴费中心传送过来的签名
+            String signatureString = requestMap.get("signatureString");
+
+            // 验签
+            boolean flag = signatureAndVerification.read_cer_and_verify_sign(requestBody,
+                    signatureString);
+
+            log.info("【QueryBill:getBill4DirectJoinMerch】缴费中心响应的报文验签结果为:{}" , flag);
+
+            QueryBillRequest queryBillRequest = JSON.parseObject(requestBodyOfDecoded,
+                    new TypeReference<QueryBillRequest>() {
+                    });
+            //交易编号
+            String traceNo = queryBillRequest.getMessage().getInfo()
+                    .getTraceNo();
+            //返回给缴费中心的响应
+            QueryBillResponse response = new QueryBillResponse(queryBillRequest);
+            QueryBillResponse.Message respMessage = response.getMessage();
+            QueryBillResponse.Message.Head respHead = response.getMessage()
+                    .getHead();
+            QueryBillResponse.Message.Info respInfo = response.getMessage()
+                    .getInfo();
+            //缴费账单子账单
+            ArrayList<QueryBillResponse.Message.Info.Bill> respBills = new ArrayList<QueryBillResponse.Message.Info.Bill>();
+            ArrayList<QueryBillResponse.Message.Info.Bill.DescDetail> respDescDetail =
+                    new ArrayList<QueryBillResponse.Message.Info.Bill.DescDetail>();
+            QueryBillResponse.Message.Info.Bill respBill = respInfo.new Bill();
+            //缴费子商户账单
+//            ArrayList<QueryBillResponse.Message.Info.Bill.SplitSubMerInfo> splitSubMerInfos = new ArrayList<QueryBillResponse.Message.Info.Bill.SplitSubMerInfo>();
+
+            //封装返回给缴费中心的响应
+
+            String epayCode = queryBillRequest.getMessage().getInfo()
+                    .getEpayCode();
+            respInfo.setEpayCode(epayCode);
+            String merchantId = queryBillRequest.getMessage().getInfo()
+                    .getMerchantId();
+            respInfo.setMerchantId(merchantId);
+            respInfo.setTraceNo(traceNo);
+            respInfo.setInput1(queryBillRequest.getMessage().getInfo()
+                    .getInput1());
+            respInfo.setInput2(queryBillRequest.getMessage().getInfo()
+                    .getInput2());
+            respInfo.setInput3(queryBillRequest.getMessage().getInfo()
+                    .getInput3());
+            respInfo.setInput4(queryBillRequest.getMessage().getInfo()
+                    .getInput4());
+            respInfo.setInput5(queryBillRequest.getMessage().getInfo()
+                    .getInput5());
+            String orderid= queryBillRequest.getMessage().getInfo().getInput1();
+            if (StringUtils.isEmpty(orderid)){
+                respHead.setReturnCode("0009");
+                respHead.setReturnMessage("参数错误,input1订单号不能为空");
+            }else{
+                if (flag){
+
+                    TPayOrder order = payOrderService.getById(orderid);
+                    List<TOrderBill> orderBills = orderBillService.getByOrderNo(orderid);
+                    List<TBillDto> bills = orderBills.stream().map(ob
+                            -> tBillService.getDetailByBillId(ob.getBillId())).collect(Collectors.toList());
+
+//	        封装详细账单信息
+                    respBill.setBillName(order.getUserName());
+                    respBill.setFeeAmt(BigDecimal.valueOf(order.getAmount()).divide(AmountConstant.b100,2, RoundingMode.HALF_DOWN).toPlainString());
+                    respBills.add(respBill);
+                    respInfo.setCustName(order.getUserName());
+                    respInfo.setCustAddress("");
+                    respInfo.setCacheMem("");
+                    respInfo.setRemark("");
+                    respInfo.setCallBackText("西藏国资委");
+                    //respInfo.setCallBackUrl("https://abcsr.keepfx.cn/b/ejy/payResult/");
+                    //使用base64加密信息
+//                    respInfo.setCallBackUrl("aHR0cDp3d3cuYWJjaGluYS5jb20vY24v");
+                    //金额规则字段
+                    String amtRule = "0";
+                    respInfo.setAmtRule(amtRule);
+	            /*QueryBillResponse.Message.Info.Bill.UnitDetail unitDetail = respBill.new UnitDetail(
+	                    "unitName", "6.66", "1");*/
+                    BigDecimal outstandAmount = bills.stream().map(TBillDto::getOutstandingMoney).reduce(BigDecimal::add).get();
+                    //欠费金额
+                    respBill.setOweAmt(outstandAmount.setScale(2,RoundingMode.HALF_DOWN).toPlainString());
+                    respBill.setFeeAmt("0.00");
+//
+//                    QueryBillResponse.Message.Info.Bill.DescDetail descDtail1 = respBill.new DescDetail(
+//                            "缴费月份:", "2020年6月份");
+//                    QueryBillResponse.Message.Info.Bill.DescDetail descDtail2 = respBill.new DescDetail(
+//                            "供电局编号:", "4340152");
+//                    QueryBillResponse.Message.Info.Bill.DescDetail descDtail3 = respBill.new DescDetail(
+//                            "欠费金额:", "0.00元");
+//                    QueryBillResponse.Message.Info.Bill.DescDetail descDtail4 = respBill.new DescDetail(
+//                            "缴费月份:", "2020年6月份");
+//                    QueryBillResponse.Message.Info.Bill.DescDetail descDtail5 = respBill.new DescDetail(
+//                            "服务时间:", "每天0:30-23:30期间均可缴费");
+//                    QueryBillResponse.Message.Info.Bill.DescDetail descDtail6 = respBill.new DescDetail(
+//                            "温馨提示:", "北京电力电费代缴,咨询电话:95598 该用户为:预付费用户");
+//                    respDescDetail.add(descDtail1);
+//                    respDescDetail.add(descDtail2);
+//                    respDescDetail.add(descDtail3);
+//                    respDescDetail.add(descDtail4);
+//                    respDescDetail.add(descDtail5);
+//                    respDescDetail.add(descDtail6);
+
+//                    respBill.setRcvMerchantId("103881104410001");
+
+//	          商户子商户详细信息
+//                    QueryBillResponse.Message.Info.Bill.SplitSubMerInfo splitSubMerInfo1 =respBill.new SplitSubMerInfo("10388", "0.01");
+//                    QueryBillResponse.Message.Info.Bill.SplitSubMerInfo splitSubMerInfo2 =respBill.new SplitSubMerInfo("1038819201", "0.02");
+//                    splitSubMerInfos.add(splitSubMerInfo1);
+//                    splitSubMerInfos.add(splitSubMerInfo2);
+
+//                    respBill.setSplitSubMerInfos(splitSubMerInfos);
+//                    respBill.setDescDetails(respDescDetail);
+                    respInfo.setTotalBillCount(String.valueOf(respBills.size()));
+                    respInfo.setBill(respBills);
+
+                    // 有定制电子回单附言信息的,需添加自定义定制附言信息字段
+                    respInfo.setMerchantRemark("");
+                    respHead.setReturnCode("0000");
+                    respHead.setReturnMessage("账单查询成功,返回成功标志");
+                }else {
+                    respHead.setReturnCode("0009");
+                    respHead.setReturnMessage("缴费中心传送给商户的请求报文签名验签失败!");
+                }
+            }
+
+            respHead.setTransFlag("02");
+            respHead.setTimeStamp(DateUtil.format(new Date(),"yyyyMMddHHmmssSSS"));
+
+            respMessage.setInfo(respInfo);
+            respMessage.setHead(respHead);
+            response.setMessage(respMessage);
+            responseJson = JSON.toJSONString(response);
+            // 加签名
+            String signatrue = signatureAndVerification
+                    .signWhithsha1withrsa(responseJson);
+            log.info("signatrue" + responseJson);
+            log.info("responseJson打印结果是(responseJson加密前):" + responseJson);
+            responseJson = signatrue + "||"
+                    + new String(Base64.encodeBase64(responseJson.getBytes("utf-8")));
+            log.info("responseJson打印结果是(responseJson加密后):{}", responseJson);
+            httpServletResponse.setCharacterEncoding("utf-8");
+            httpServletResponse.setContentType("text/plain");
+            httpServletResponse.getWriter().write(responseJson);
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+    }
 
 
 }
diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/COSController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/COSController.java
index 0702deb..6aa9856 100644
--- a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/COSController.java
+++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/COSController.java
@@ -1,19 +1,33 @@
 package com.ruoyi.web.controller.api;
 
+import com.alibaba.fastjson2.JSON;
+import com.ruoyi.common.config.FileUploadConfig;
+import com.ruoyi.common.constant.CacheConstants;
 import com.ruoyi.common.core.domain.R;
+import com.ruoyi.common.core.redis.RedisCache;
+import com.ruoyi.common.utils.StringUtils;
+import com.ruoyi.framework.web.service.TokenService;
+import com.ruoyi.system.model.TFile;
+import com.ruoyi.system.service.SysFileService;
+import com.ruoyi.system.service.impl.SysFileServiceImpl;
 import com.ruoyi.web.controller.tool.TencentCosUtil;
 import io.swagger.annotations.Api;
 import io.swagger.annotations.ApiImplicitParam;
 import io.swagger.annotations.ApiImplicitParams;
 import io.swagger.annotations.ApiOperation;
-import org.springframework.web.bind.annotation.PostMapping;
-import org.springframework.web.bind.annotation.RequestMapping;
-import org.springframework.web.bind.annotation.RequestParam;
-import org.springframework.web.bind.annotation.RestController;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.*;
+import org.springframework.web.context.request.RequestContextHolder;
+import org.springframework.web.context.request.ServletRequestAttributes;
 import org.springframework.web.multipart.MultipartFile;
 
 import javax.annotation.Resource;
-import java.io.OutputStream;
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.io.IOException;
+import java.io.PrintWriter;
+import java.net.URLEncoder;
 
 /**
  * @author HJL
@@ -23,18 +37,145 @@
 @RestController
 @RequestMapping("/cos")
 @Api(tags = "公共-文件上传")
+@Slf4j
 public class COSController {
 
     @Resource
     private TencentCosUtil tencentCosUtil;
 
+    @Autowired
+    SysFileService sysFileService;
+
+    @Autowired
+    FileUploadConfig fileUploadConfig;
+
+    @Autowired
+    TokenService tokenService;
+
+    public String getLocalUrlPrefix(){
+        ServletRequestAttributes servletRequestAttributes = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes());
+        if (servletRequestAttributes==null || servletRequestAttributes.getRequest()==null){
+            return fileUploadConfig.getFileUrlPrefix();
+        }
+        HttpServletRequest request = servletRequestAttributes.getRequest();
+        StringBuffer url = new StringBuffer();
+        url.append(request.getScheme()).append("://")
+                .append(request.getServerName())
+                .append((request.getServerPort() == 80 ? "" : ":" + request.getServerPort()))
+                .append(request.getContextPath());
+        return url.toString();
+    }
+
+    public  String getLocalFileUrlPrefix(String fileId){
+        String token = tokenService.getLoginUser().getToken();
+        StringBuffer url = new StringBuffer();
+        url.append(getLocalUrlPrefix())
+                .append("/cos/get/").append(fileId).append("?s=").append(URLEncoder.encode(token))
+        ;
+        return url.toString();
+    }
+
+
+
+
+
+
+    public static void failResponse(String message){
+        ServletRequestAttributes servletRequestAttributes = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes());
+        HttpServletResponse response = servletRequestAttributes.getResponse();
+        String failResult =  JSON.toJSONString(R.fail(message));
+        //设置编码格式
+        response.setCharacterEncoding("UTF-8");
+        response.setContentType("application/json;charset=UTF-8");
+        PrintWriter pw = null;
+        try {
+            pw = response.getWriter();
+            pw.write(failResult);
+            pw.flush();
+        } catch (IOException e) {
+            log.error("io异常");
+        }finally {
+            if (pw!=null) {
+                pw.close();
+            }
+        }
+
+    }
+    @Autowired
+    private RedisCache redisCache;
+
+    @GetMapping("get/{fileId}")
+    public void get(@PathVariable("fileId") String fileid,@RequestParam("s") String s){
+        if (StringUtils.isEmpty(fileid)){
+            failResponse("文件ID不能为空");
+            return;
+        }
+        if (StringUtils.isEmpty(s)){
+            failResponse("token不能为空");
+            return;
+        }
+        Object object = redisCache.getCacheObject(CacheConstants.LOGIN_TOKEN_KEY + s);
+        if (object==null){
+            failResponse("用户登录已失效");
+            return;
+        }
+        TFile file = sysFileService.getById(fileid);
+        if (file==null){
+            failResponse("图片不存在");
+            return;
+        }
+        tencentCosUtil.download(file);
+    }
+
+
+    @GetMapping("get/file")
+    public void getFile(@RequestParam("fileUrl") String fileUrl,@RequestParam("s") String s){
+        if (StringUtils.isEmpty(fileUrl)){
+            failResponse("文件路径不能为空");
+            return;
+        }
+        if (StringUtils.isEmpty(s)){
+            failResponse("token不能为空");
+            return;
+        }
+        Object object = redisCache.getCacheObject(CacheConstants.LOGIN_TOKEN_KEY + s);
+        if (object==null){
+            failResponse("用户登录已失效");
+            return;
+        }
+        tencentCosUtil.download(fileUrl);
+    }
+
+    /**
+     * 新上传接口,下一版更新
+     * @param file
+     * @param folder 上传到cos的文件目录:如/contract/
+     * @return
+     */
+    @PostMapping("/uploadnew")
+    @ApiOperation(value = "文件上传,带上传目录,返回文件ID", tags = "公共-文件上传")
+    @ApiImplicitParams({
+            @ApiImplicitParam(value = "文件", name = "file", dataType = "MultipartFile", required = true)
+    })
+    public R<TFile> uploadnew(@RequestParam("file") MultipartFile file, @RequestParam("folder") String folder) {
+        TFile tFile = tencentCosUtil.upload(file,folder);
+        tFile.setFileUrl(getLocalFileUrlPrefix(tFile.getId()));
+        return R.ok(tFile);
+    }
+
+    /**
+     *
+     * @param file
+     * @param
+     * @return
+     */
     @PostMapping("/upload")
     @ApiOperation(value = "文件上传", tags = "公共-文件上传")
     @ApiImplicitParams({
             @ApiImplicitParam(value = "文件", name = "file", dataType = "MultipartFile", required = true)
     })
-    public R<String> upload(@RequestParam("file") MultipartFile file) {
-        String url = tencentCosUtil.upLoadFile(file);
+    public R<String> upload(@RequestParam("file") MultipartFile file,@RequestParam("folder") String folder) {
+        String url = tencentCosUtil.upLoadFile(file,folder);
         return R.ok(url, url);
     }
     @PostMapping("/downloadImg")
diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/SysFileController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/SysFileController.java
new file mode 100644
index 0000000..b213e84
--- /dev/null
+++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/SysFileController.java
@@ -0,0 +1,22 @@
+package com.ruoyi.web.controller.api;
+
+
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+/**
+ * <p>
+ * 文件列表 前端控制器
+ * </p>
+ *
+ * @author yupeng
+ * @since 2025-03-05
+ */
+@RestController
+@RequestMapping("/sys-file")
+public class SysFileController {
+
+
+
+}
+
diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TContractController.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TContractController.java
index e38938d..bdd80b0 100644
--- a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TContractController.java
+++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/api/TContractController.java
@@ -331,6 +331,7 @@
 
             }
             String url = wordUtil.generatePdf("/template", "1_yzj_租赁合同.xml", templateParam, "租赁合同", "E:\\");
+            System.out.println(url);
             res.add(url);
         }
 
diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/PdfUtils.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/PdfUtils.java
index 71c5a6c..c786d9b 100644
--- a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/PdfUtils.java
+++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/PdfUtils.java
@@ -50,7 +50,7 @@
             //上传图片
             byte2File(stream.toByteArray(),filePath + "/pdf",fileName.substring(0,fileName.lastIndexOf(".")) + ".pdf");
             MultipartFile multipartFile = convertToMultipartFile(stream,fileName.substring(0,fileName.lastIndexOf(".")) );
-            String s = tencentCosUtil.upLoadFile(multipartFile);
+            String s = tencentCosUtil.upLoadFile(multipartFile,"/wordToPdf");
 
             stream.close();
             inputStream.close();
diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/TencentCosUtil.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/TencentCosUtil.java
index 1d02134..3834733 100644
--- a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/TencentCosUtil.java
+++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/TencentCosUtil.java
@@ -1,20 +1,23 @@
 package com.ruoyi.web.controller.tool;
 
+import cn.hutool.core.date.DateUtil;
 import com.qcloud.cos.COSClient;
 import com.qcloud.cos.ClientConfig;
 import com.qcloud.cos.auth.BasicCOSCredentials;
 import com.qcloud.cos.auth.COSCredentials;
-import com.qcloud.cos.exception.CosClientException;
-import com.qcloud.cos.exception.CosServiceException;
 import com.qcloud.cos.http.HttpProtocol;
 import com.qcloud.cos.model.COSObject;
-import com.qcloud.cos.model.GetObjectRequest;
 import com.qcloud.cos.model.ObjectMetadata;
 import com.qcloud.cos.model.PutObjectResult;
 import com.qcloud.cos.region.Region;
-import com.qcloud.cos.utils.IOUtils;
 import com.ruoyi.common.utils.WebUtils;
-import org.springframework.beans.factory.annotation.Value;
+import com.ruoyi.system.model.TFile;
+import com.ruoyi.system.service.SysFileService;
+import com.ruoyi.system.service.impl.SysFileServiceImpl;
+import com.ruoyi.web.core.config.FileUploaderConfig;
+import lombok.extern.slf4j.Slf4j;
+import org.apache.commons.lang3.StringUtils;
+import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.stereotype.Component;
 import org.springframework.web.multipart.MultipartFile;
 
@@ -25,44 +28,17 @@
 import java.io.InputStream;
 import java.io.OutputStream;
 import java.util.Base64;
+import java.util.Date;
 import java.util.UUID;
+
+import static cn.hutool.core.date.DatePattern.NORM_DATE_FORMAT;
 
 /**
  * @author HJL
  */
 @Component
+@Slf4j
 public class TencentCosUtil {
-
-    /**
-     * COS的SecretId
-     */
-    @Value("${cos.client.accessKey}")
-    private String secretId;
-    /**
-     * COS的SecretKey
-     */
-    @Value("${cos.client.secretKey}")
-    private String secretKey;
-    /**
-     * 文件上传后访问路径的根路径,后面要最佳文件名字与类型
-     */
-    @Value("${cos.client.rootSrc}")
-    private String rootSrc;
-    /**
-     * 上传的存储桶的地域
-     */
-    @Value("${cos.client.bucketAddr}")
-    private String bucketAddr;
-    /**
-     * 存储桶的名字,是自己在存储空间自己创建的,我创建的名字是:qq-test-1303******
-     */
-    @Value("${cos.client.bucket}")
-    private String bucketName;
-    /**
-     * 文件存放位置
-     */
-    @Value("${cos.client.location}")
-    private String location;
 
     /**
      * 1.调用静态方法getCosClient()就会获得COSClient实例
@@ -70,16 +46,64 @@
      *
      * @return COSClient实例
      */
-    private COSClient getCosClient() {
-        // 1 初始化用户身份信息(secretId, secretKey)。
-        COSCredentials cred = new BasicCOSCredentials(secretId, secretKey);
-        // 2.1 设置存储桶的地域(上文获得)
-        Region region = new Region(bucketAddr);
-        ClientConfig clientConfig = new ClientConfig(region);
-        // 2.2 使用https协议传输
-        clientConfig.setHttpProtocol(HttpProtocol.https);
-        // 生成 cos 客户端
-        return new COSClient(cred, clientConfig);
+
+    @Autowired
+    COSClient cosClient;
+
+    @Autowired
+    FileUploaderConfig cosConfig;
+
+    @Autowired
+    SysFileService sysFileService;
+
+
+    /**
+     * 上传文件,并存入sys_file,返回文件的主键ID
+     * @param file
+     * @param folder  格式:/xxxxx,/xxxx/xxxx ,最后不用加斜杠
+     * @return
+     */
+    public TFile upload(MultipartFile file, String folder){
+        try {
+            // 获取上传的文件的输入流
+            InputStream inputStream = file.getInputStream();
+            // 避免文件覆盖,获取文件的原始名称,如123.jpg,然后通过截取获得文件的后缀,也就是文件的类型
+
+            String originalFilename = file.getOriginalFilename();
+
+            //获取文件的类型
+            String fileType = originalFilename.substring(originalFilename.lastIndexOf("."));
+            //使用UUID工具  创建唯一名称,放置文件重名被覆盖,在拼接上上命令获取的文件类型
+            String fileName = UUID.randomUUID() + fileType;
+            String filePath = (StringUtils.isNotEmpty(folder)?
+                    folder+"/"
+                    :"/default/") + DateUtil.format(new Date(),NORM_DATE_FORMAT)+"/"+fileName;
+            // 指定文件上传到 COS 上的路径,即对象键。最终文件会传到存储桶名字中的images文件夹下的fileName名字
+            filePath = cosConfig.getLocation() + filePath;
+            // 创建上传Object的Metadata
+            ObjectMetadata objectMetadata = new ObjectMetadata();
+            // - 使用输入流存储,需要设置请求长度
+            objectMetadata.setContentLength(inputStream.available());
+            // - 设置缓存
+            objectMetadata.setCacheControl("no-cache");
+            // - 设置Content-Type
+            objectMetadata.setContentType(fileType);
+            //上传文件
+            cosClient.putObject(cosConfig.getBucketName(), filePath, inputStream, objectMetadata);
+            TFile tFile = new TFile();
+            tFile.setFileName(filePath);
+            tFile.setRealName(originalFilename);
+            tFile.setFileType(fileType);
+            tFile.setUrl(cosConfig.getRootSrc()+filePath);
+            tFile.setContentType(file.getContentType());
+            tFile.setFileSize(file.getSize());
+            sysFileService.save(tFile);
+            return tFile;
+        } catch (Exception e) {
+            log.error("上传文件发生异常",e);
+            // 发生IO异常、COS连接异常等,返回空
+            return null;
+        }
     }
 
     /**
@@ -88,7 +112,7 @@
      * @param file
      * @return 返回文件的浏览全路径
      */
-    public String upLoadFile(MultipartFile file) {
+    public String upLoadFile(MultipartFile file,String folder) {
         try {
             // 获取上传的文件的输入流
             InputStream inputStream = file.getInputStream();
@@ -99,7 +123,11 @@
             //使用UUID工具  创建唯一名称,放置文件重名被覆盖,在拼接上上命令获取的文件类型
             String fileName = UUID.randomUUID() + fileType;
             // 指定文件上传到 COS 上的路径,即对象键。最终文件会传到存储桶名字中的images文件夹下的fileName名字
-            String key = location+"/" + fileName;
+            String filePath = (StringUtils.isNotEmpty(folder)?
+                    folder+"/"
+                    :"/default/") + DateUtil.format(new Date(),NORM_DATE_FORMAT)+"/"+fileName;
+
+            filePath = cosConfig.getLocation()+"/" + filePath;
             // 创建上传Object的Metadata
             ObjectMetadata objectMetadata = new ObjectMetadata();
             // - 使用输入流存储,需要设置请求长度
@@ -109,14 +137,13 @@
             // - 设置Content-Type
             objectMetadata.setContentType(fileType);
             //上传文件
-            PutObjectResult putResult = getCosClient().putObject(bucketName, key, inputStream, objectMetadata);
+            PutObjectResult putResult = cosClient.putObject(cosConfig.getBucketName(), filePath, inputStream, objectMetadata);
             // 创建文件的网络访问路径
-            String url = rootSrc + key;
-            //关闭 cosClient,并释放 HTTP 连接的后台管理线程
-            getCosClient().shutdown();
+            String url = cosConfig.getRootSrc() + filePath;
             return url;
         } catch (Exception e) {
             e.printStackTrace();
+            log.error("上传文件发生异常",e);
             // 发生IO异常、COS连接异常等,返回空
             return null;
         }
@@ -129,20 +156,11 @@
      */
     public void downLoadFile(String file) {
         HttpServletResponse response = WebUtils.response();
-        String replace = file.replace(rootSrc, "");
+        String replace = file.replace(cosConfig.getRootSrc(), "");
         response.setHeader("Access-Control-Expose-Headers","File-Type");
-        COSCredentials cred = new BasicCOSCredentials(
-                secretId,
-                secretKey);
-        // 2.1 设置存储桶的地域(上文获得)
-        Region region = new Region(bucketAddr);
-        ClientConfig clientConfig = new ClientConfig(region);
-        // 2.2 使用https协议传输
-        clientConfig.setHttpProtocol(HttpProtocol.https);
-        COSClient cosClient = new COSClient(cred, clientConfig);
         try {
             // 5. 下载文件并获取输入流
-            InputStream inputStream = cosClient.getObject(bucketName, replace).getObjectContent();
+            InputStream inputStream = cosClient.getObject(cosConfig.getBucketName(), replace).getObjectContent();
             ServletOutputStream outputStream = response.getOutputStream();
             // 6. 处理输入流,例如读取内容或保存到本地文件
             // 这里仅作示例,实际应用中需要根据需求处理输入流
@@ -154,26 +172,15 @@
             }
         } catch (Exception e) {
             e.printStackTrace();
-        } finally {
-            // 7. 关闭输入流
-            cosClient.shutdown();
+            log.error("下载文件发生异常",e);
         }
     }
     public String downLoadFileImg(String file) {
         byte[] data = null;
-        String replace = file.replace(rootSrc, "");
-        COSCredentials cred = new BasicCOSCredentials(
-                secretId,
-                secretKey);
-        // 2.1 设置存储桶的地域(上文获得)
-        Region region = new Region(bucketAddr);
-        ClientConfig clientConfig = new ClientConfig(region);
-        // 2.2 使用https协议传输
-        clientConfig.setHttpProtocol(HttpProtocol.https);
-        COSClient cosClient = new COSClient(cred, clientConfig);
+        String replace = file.replace(cosConfig.getRootSrc(), "");
         try {
             // 5. 下载文件并获取输入流
-            InputStream inputStream = cosClient.getObject(bucketName, replace).getObjectContent();
+            InputStream inputStream = cosClient.getObject(cosConfig.getBucketName(), replace).getObjectContent();
             ByteArrayOutputStream swapStream = new ByteArrayOutputStream();
             // 6. 处理输入流,例如读取内容或保存到本地文件
             byte[] buffer = new byte[1024];
@@ -185,10 +192,60 @@
             data = swapStream.toByteArray();
         } catch (Exception e) {
             e.printStackTrace();
+            log.error("下载图片发生异常",e);
         } finally {
-            // 7. 关闭输入流
-            cosClient.shutdown();
         }
         return Base64.getEncoder().encodeToString(data);
     }
+
+    public void download(String fileUrl) {
+        HttpServletResponse response = WebUtils.response();
+        fileUrl = fileUrl.replace(cosConfig.getRootSrc(), "");
+
+            // 5. 下载文件并获取输入流
+        COSObject object = cosClient.getObject(cosConfig.getBucketName(),fileUrl);
+        try (
+            InputStream is = object.getObjectContent();
+             OutputStream os = response.getOutputStream()
+        ) {
+            String fileName = fileUrl.substring(fileUrl.lastIndexOf("/"));
+            response.setContentType(object.getObjectMetadata().getContentType() + ";charset=utf-8");
+            String filename = new String(fileName.getBytes("UTF-8"), "ISO-8859-1");
+            response.addHeader("Content-Disposition", "attachment;filename=" + filename);
+            response.addHeader("Content-Length", "" + object.getObjectMetadata().getContentLength());
+            int len = 0;
+            byte[] buffer = new byte[2048];
+            while ((len = is.read(buffer)) > 0) {
+                os.write(buffer, 0, len);
+            }
+            os.flush();
+        } catch (IOException e) {
+            log.error("读取cos图片发生异常", e);
+        }
+    }
+
+    public void download(TFile file) {
+        HttpServletResponse response = WebUtils.response();
+        // 5. 下载文件并获取输入流
+        COSObject object = cosClient.getObject(cosConfig.getBucketName(), file.getFileName());
+        try (
+                InputStream is = object.getObjectContent();
+                OutputStream os = response.getOutputStream()
+        ) {
+            response.setContentType(file.getContentType() + ";charset=utf-8");
+            String filename = new String(file.getRealName().getBytes("UTF-8"), "ISO-8859-1");
+            response.addHeader("Content-Disposition", "attachment;filename=" + filename);
+            response.addHeader("Content-Length", "" + file.getFileSize());
+            int len = 0;
+            byte[] buffer = new byte[2048];
+            while ((len = is.read(buffer)) > 0) {
+                os.write(buffer, 0, len);
+            }
+            os.flush();
+        } catch (IOException e) {
+            log.error("读取cos图片发生异常", e);
+        }
+    }
+
+
 }
\ No newline at end of file
diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/WordUtil.java b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/WordUtil.java
index 85a55ce..ce14b96 100644
--- a/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/WordUtil.java
+++ b/ruoyi-admin/src/main/java/com/ruoyi/web/controller/tool/WordUtil.java
@@ -101,7 +101,7 @@
                 fis.read(fileContent);
             }
             MultipartFile mockMultipartFile = new MockMultipartFile(encodedFileName+".doc", fileContent);
-            String s = tencentCosUtil.upLoadFile(mockMultipartFile);
+            String s = tencentCosUtil.upLoadFile(mockMultipartFile,"/wordGenerate");
             return s;
         } catch (IOException | TemplateException e) {
             log.error("生成Word文档异常,异常原因:{}", e.getMessage(), e);
diff --git a/ruoyi-admin/src/main/java/com/ruoyi/web/core/config/FileUploaderConfig.java b/ruoyi-admin/src/main/java/com/ruoyi/web/core/config/FileUploaderConfig.java
new file mode 100644
index 0000000..d87ab0d
--- /dev/null
+++ b/ruoyi-admin/src/main/java/com/ruoyi/web/core/config/FileUploaderConfig.java
@@ -0,0 +1,67 @@
+package com.ruoyi.web.core.config;
+
+import com.qcloud.cos.COSClient;
+import com.qcloud.cos.ClientConfig;
+import com.qcloud.cos.auth.BasicCOSCredentials;
+import com.qcloud.cos.auth.COSCredentials;
+import com.qcloud.cos.http.HttpProtocol;
+import com.qcloud.cos.region.Region;
+import lombok.Data;
+import org.springframework.beans.factory.annotation.Value;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+
+@Configuration
+@Data
+public class FileUploaderConfig {
+
+    /**
+     * COS的SecretId
+     */
+    @Value("${cos.client.accessKey}")
+    private String secretId;
+    /**
+     * COS的SecretKey
+     */
+    @Value("${cos.client.secretKey}")
+    private String secretKey;
+    /**
+     * 文件上传后访问路径的根路径,后面要最佳文件名字与类型
+     */
+    @Value("${cos.client.rootSrc}")
+    private String rootSrc;
+    /**
+     * 上传的存储桶的地域
+     */
+    @Value("${cos.client.bucketAddr}")
+    private String bucketAddr;
+    /**
+     * 存储桶的名字,是自己在存储空间自己创建的,我创建的名字是:qq-test-1303******
+     */
+    @Value("${cos.client.bucket}")
+    private String bucketName;
+    /**
+     * 文件存放位置
+     */
+    @Value("${cos.client.location}")
+    private String location;
+
+    @Value("${file.url.prefix}")
+    private String fileUrlPrefix;
+
+
+    @Bean
+    public COSClient cosClient() {
+        // 1 初始化用户身份信息(secretId, secretKey)。
+        COSCredentials cred = new BasicCOSCredentials(secretId, secretKey);
+        // 2.1 设置存储桶的地域(上文获得)
+        Region region = new Region(bucketAddr);
+        ClientConfig clientConfig = new ClientConfig(region);
+        // 2.2 使用https协议传输
+        clientConfig.setHttpProtocol(HttpProtocol.https);
+        // 生成 cos 客户端
+        return new COSClient(cred, clientConfig);
+    }
+
+
+}
diff --git a/ruoyi-admin/src/main/resources/application-prod.yml b/ruoyi-admin/src/main/resources/application-prod.yml
index caf8b52..99a087e 100644
--- a/ruoyi-admin/src/main/resources/application-prod.yml
+++ b/ruoyi-admin/src/main/resources/application-prod.yml
@@ -213,4 +213,4 @@
     bucket: xzgttest-1305134071
     bucketAddr: ap-chengdu
     rootSrc: https://xzgttest-1305134071.cos.ap-chengdu.myqcloud.com/
-    location: xizang
\ No newline at end of file
+    location: /xizang
\ No newline at end of file
diff --git a/ruoyi-admin/src/main/resources/application-test.yml b/ruoyi-admin/src/main/resources/application-test.yml
index ec0ed40..4fa25ea 100644
--- a/ruoyi-admin/src/main/resources/application-test.yml
+++ b/ruoyi-admin/src/main/resources/application-test.yml
@@ -14,7 +14,6 @@
   addressEnabled: false
   # 验证码类型 math 数字计算 char 字符验证
   captchaType: math
-
 # 开发环境配置
 server:
   # 服务器的HTTP端口,默认为8080
@@ -103,7 +102,7 @@
     druid:
       # 主库数据源
       master:
-        url: jdbc:mysql://xzgt.test.591taxi.cn:13306/xizang?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai
+        url: jdbc:mysql://xzgt.test.591taxi.cn:13306/xizang?autoReconnect=true&useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai
         username: root
         password: 8f5z9g52gx4bg
       # 从库数据源
@@ -199,6 +198,8 @@
     qrLocation: /file/qrCode/
     accessPath: /file/
     allowExt: .jpg|.png|.gif|.jpeg|.doc|.docx|.apk|.MP4|.mp4|.pdf|.PDF
+  url:
+    prefix: http://localhost:${server.port}${server.servlet.context-path}
 wx:
   conf:
     appId: wxe91f1af7638aa5dd
@@ -219,10 +220,18 @@
     bucket: xzgttest-1305134071
     bucketAddr: ap-chengdu
     rootSrc: https://xzgttest-1305134071.cos.ap-chengdu.myqcloud.com/
-    location: xizang
+    location: /xizang
 sms:
   enable: true
   appId: 1400957506
   secretid: AKIDCF5EF2c0DE1e5JK8r4EGJF4mNsMgp26x
   secretkey: lLl184rUyFOOE0d5KNGC3kmfNsCWk4GU
-  sign: 畅云出行
\ No newline at end of file
+  sign: 畅云出行
+com:
+  taxi591:
+    bank:
+      cer-path: D:\workspaces\工作文件\畅云\农业银行\TrustPayTest.cer
+      base-url: http://hello.enjoy.abchina.com
+      enable: true
+      keystore-password:
+      pfx-path: D:\workspaces\工作文件\畅云\农业银行\103882200000958.pfx
\ No newline at end of file
diff --git a/ruoyi-applet/src/main/java/com/ruoyi/web/controller/api/TInformationController.java b/ruoyi-applet/src/main/java/com/ruoyi/web/controller/api/TInformationController.java
index a46014a..72980d2 100644
--- a/ruoyi-applet/src/main/java/com/ruoyi/web/controller/api/TInformationController.java
+++ b/ruoyi-applet/src/main/java/com/ruoyi/web/controller/api/TInformationController.java
@@ -51,7 +51,9 @@
     public R<TInformation> getDetailById(@RequestParam String id) {
         // 处理查看次数
         redisCache.increment(Constants.INFORMATION_VIEW + id);
-        return R.ok(informationService.getById(id));
+        TInformation information = informationService.getById(id);
+        information.setViewCount(redisCache.getCacheObject(Constants.INFORMATION_VIEW + information.getId()));
+        return R.ok(information);
     }
 
 }
diff --git a/ruoyi-applet/src/main/resources/application-test.yml b/ruoyi-applet/src/main/resources/application-test.yml
index e9018b1..3bfc62b 100644
--- a/ruoyi-applet/src/main/resources/application-test.yml
+++ b/ruoyi-applet/src/main/resources/application-test.yml
@@ -103,7 +103,7 @@
     druid:
       # 主库数据源
       master:
-        url: jdbc:mysql://xzgt.test.591taxi.cn:13306/xizang?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai
+        url: jdbc:mysql://xzgt.test.591taxi.cn:13306/xizang?autoReconnect=true&useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai
         username: root
         password: 8f5z9g52gx4bg
       # 从库数据源
diff --git a/ruoyi-common/src/main/java/com/ruoyi/common/config/FileUploadConfig.java b/ruoyi-common/src/main/java/com/ruoyi/common/config/FileUploadConfig.java
index 3a8b0ed..32c0b2a 100644
--- a/ruoyi-common/src/main/java/com/ruoyi/common/config/FileUploadConfig.java
+++ b/ruoyi-common/src/main/java/com/ruoyi/common/config/FileUploadConfig.java
@@ -18,4 +18,6 @@
     private String allowExt;
     private String location;
     private String qrLocation;
+    private String fileUrlPrefix;
+
 }
diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/SecurityConfig.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/SecurityConfig.java
index 5f90047..b4118dc 100644
--- a/ruoyi-framework/src/main/java/com/ruoyi/framework/config/SecurityConfig.java
+++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/config/SecurityConfig.java
@@ -119,7 +119,7 @@
                         "/operations/getBySingleNum/**",
                         "/user/getUserInfoByNumber/**",
                         "/wxLogin/**",
-                        "/open/**"
+                        "/open/**","/cos/get/**"
                 ).permitAll()
                 // 静态资源,可匿名访问
                 .antMatchers(HttpMethod.GET, "/", "/*.html", "/**/*.html", "/**/*.css", "/**/*.js", "/profile/**").permitAll()
diff --git a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/TokenService.java b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/TokenService.java
index a7a640b..206c1a4 100644
--- a/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/TokenService.java
+++ b/ruoyi-framework/src/main/java/com/ruoyi/framework/web/service/TokenService.java
@@ -204,6 +204,14 @@
             refreshToken(loginUser);
         }
     }
+
+
+    public boolean verifyToken(String token)
+    {
+        Claims claims = parseToken(token);
+
+        return true;
+    }
     /**
      * 小程序验证令牌有效期,相差不足20分钟,自动刷新缓存
      *
diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysFileMapper.java b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysFileMapper.java
new file mode 100644
index 0000000..f5e0298
--- /dev/null
+++ b/ruoyi-system/src/main/java/com/ruoyi/system/mapper/SysFileMapper.java
@@ -0,0 +1,16 @@
+package com.ruoyi.system.mapper;
+
+import com.baomidou.mybatisplus.core.mapper.BaseMapper;
+import com.ruoyi.system.model.TFile;
+
+/**
+ * <p>
+ * 文件列表 Mapper 接口
+ * </p>
+ *
+ * @author yupeng
+ * @since 2025-03-05
+ */
+public interface SysFileMapper extends BaseMapper<TFile> {
+
+}
diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/model/TContract.java b/ruoyi-system/src/main/java/com/ruoyi/system/model/TContract.java
index c2c2834..7524013 100644
--- a/ruoyi-system/src/main/java/com/ruoyi/system/model/TContract.java
+++ b/ruoyi-system/src/main/java/com/ruoyi/system/model/TContract.java
@@ -172,6 +172,6 @@
     private String houseAddress;
     @ApiModelProperty(value = "审批流实例id")
     @TableField(exist = false)
-    private Long instanceId;
+    private String instanceId;
 
 }
diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/model/TFile.java b/ruoyi-system/src/main/java/com/ruoyi/system/model/TFile.java
new file mode 100644
index 0000000..93be19a
--- /dev/null
+++ b/ruoyi-system/src/main/java/com/ruoyi/system/model/TFile.java
@@ -0,0 +1,81 @@
+package com.ruoyi.system.model;
+
+import com.baomidou.mybatisplus.annotation.TableField;
+import com.baomidou.mybatisplus.annotation.TableId;
+import com.baomidou.mybatisplus.annotation.TableName;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+
+import java.io.Serializable;
+import java.time.LocalDateTime;
+
+/**
+ * <p>
+ * 文件列表
+ * </p>
+ *
+ * @author yupeng
+ * @since 2025-03-05
+ */
+@Data
+@EqualsAndHashCode(callSuper = false)
+@TableName("t_file")
+@ApiModel(value="t_file对象", description="文件列表")
+public class TFile implements Serializable {
+
+    private static final long serialVersionUID = 1L;
+
+    @TableId("id")
+    private String id;
+
+    @ApiModelProperty(value = "文件分类,0.用户文件 1.钉钉文件")
+    @TableField("type")
+    private Integer type;
+
+    @TableField("content_type")
+    private String contentType;
+
+    @ApiModelProperty(value = "文件类型(后缀)")
+    @TableField("file_type")
+    private String fileType;
+
+    @ApiModelProperty(value = "文件真实名称")
+    @TableField("real_name")
+    private String realName;
+
+    @ApiModelProperty(value = "文件名称")
+    @TableField("file_name")
+    private String fileName;
+
+    @ApiModelProperty(value = "文件大小")
+    @TableField("file_size")
+    private Long fileSize;
+
+    @TableField("create_time")
+    private LocalDateTime createTime;
+
+    @ApiModelProperty(value = "地址路径或者url")
+    @TableField("url")
+    private String url;
+
+    @ApiModelProperty(value = "上传人ID")
+    @TableField("creator_id")
+    private String creatorId;
+
+    @ApiModelProperty(value = "是否需要校验权限")
+    @TableField("need_auth")
+    private Boolean needAuth;
+
+    @ApiModelProperty(value = "是否有效")
+    @TableField("valid")
+    private Boolean valid;
+    /**
+     * 返回文件对象时自动组装文件访问路径
+     */
+    @TableField(exist = false)
+    @ApiModelProperty("文件访问路径")
+    private String fileUrl;
+
+}
diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/SysFileService.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/SysFileService.java
new file mode 100644
index 0000000..63b9bce
--- /dev/null
+++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/SysFileService.java
@@ -0,0 +1,17 @@
+package com.ruoyi.system.service;
+
+import com.baomidou.mybatisplus.extension.service.IService;
+import com.ruoyi.system.model.TFile;
+
+/**
+ * <p>
+ * 文件列表 服务类
+ * </p>
+ *
+ * @author yupeng
+ * @since 2025-03-05
+ */
+public interface SysFileService extends IService<TFile> {
+
+
+}
diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysFileServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysFileServiceImpl.java
new file mode 100644
index 0000000..5cd94fd
--- /dev/null
+++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/SysFileServiceImpl.java
@@ -0,0 +1,42 @@
+package com.ruoyi.system.service.impl;
+
+import com.alibaba.fastjson2.JSON;
+import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
+import com.ruoyi.common.config.FileUploadConfig;
+import com.ruoyi.common.core.domain.R;
+import com.ruoyi.system.mapper.SysFileMapper;
+import com.ruoyi.system.model.TFile;
+import com.ruoyi.system.service.SysFileService;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.security.core.token.TokenService;
+import org.springframework.stereotype.Service;
+import org.springframework.web.context.request.RequestContextHolder;
+import org.springframework.web.context.request.ServletRequestAttributes;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.io.IOException;
+import java.io.PrintWriter;
+import java.net.URLEncoder;
+
+/**
+ * <p>
+ * 文件列表 服务实现类
+ * </p>
+ *
+ * @author yupeng
+ * @since 2025-03-05
+ */
+@Service
+@Slf4j
+public class SysFileServiceImpl extends ServiceImpl<SysFileMapper, TFile> implements SysFileService {
+
+    @Autowired
+    FileUploadConfig fileUploadConfig;
+
+
+
+
+
+}
diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java
index e70e065..362d630 100644
--- a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java
+++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TBillServiceImpl.java
@@ -1,5 +1,6 @@
 package com.ruoyi.system.service.impl;
 
+import cn.hutool.core.date.DateUtil;
 import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
 import com.ruoyi.common.basic.PageInfo;
 import com.ruoyi.common.config.SmsProperties;
@@ -7,10 +8,7 @@
 import com.ruoyi.common.constant.CacheConstants;
 import com.ruoyi.common.core.redis.RedisCache;
 import com.ruoyi.common.exception.ServiceException;
-import com.ruoyi.common.utils.DateUtils;
-import com.ruoyi.common.utils.SmsUtil;
-import com.ruoyi.common.utils.StringUtils;
-import com.ruoyi.common.utils.TencentMailUtil;
+import com.ruoyi.common.utils.*;
 import com.ruoyi.common.utils.uuid.UUID;
 import com.ruoyi.system.dto.*;
 import com.ruoyi.system.mapper.TBillMapper;
@@ -252,7 +250,7 @@
             throw new ServiceException("实付金额不能高于于流水可抵扣剩余金额");
         }
         //如果实付金额大于欠费金额
-        if (dto.getAmount().compareTo(bill.getOutstandingMoney())>=0){
+        if (dto.getAmount().compareTo(bill.getOutstandingMoney())>0){
             throw new ServiceException("实付金额不能高于该账单欠费金额");
         }
 
@@ -276,6 +274,7 @@
         save.setPayType(3);
         save.setPayer(dto.getPayer());
         save.setPayTime(bankflow.getPayTime());
+        save.setSysSerialNumber(OrderNos.getDid(30));
         save.setBankSerialNumber(bankflow.getBankSerialNumber());
         save.setFlowType(2);
         save.setPaymentBillId(back.getId());
@@ -332,6 +331,7 @@
                 saveFlow.setPayType(1);
                 saveFlow.setPayer(order.getUserId());
                 saveFlow.setPayTime(DateUtils.dateToLocalDateTime(save.getPayTime()));
+                saveFlow.setSysSerialNumber(OrderNos.getDid(30));
                 saveFlow.setBankSerialNumber(save.getPayNo());
                 saveFlow.setFlowType(2);
                 saveFlow.setPaymentBillId(bill.getId());
@@ -464,7 +464,7 @@
                 throw new ServiceException("实付金额不能高于于流水可抵扣剩余金额");
             }
             //如果实付金额大于欠费金额
-            if (dto.getAmount().compareTo(bill.getOutstandingMoney())>=0){
+            if (dto.getAmount().compareTo(bill.getOutstandingMoney())>0){
                 throw new ServiceException("实付金额不能高于该账单欠费金额");
             }
         }
diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TContractServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TContractServiceImpl.java
index 0f92c56..30368c2 100644
--- a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TContractServiceImpl.java
+++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TContractServiceImpl.java
@@ -73,6 +73,12 @@
         for (TContract tContract : list) {
             tContract.setPayType(DictUtils.getDictLabel(DictConstants.DICT_TYPE_CONTRACT_PAY_TYPE,tContract.getPayType()));
             tContract.setStatus(DictUtils.getDictLabel(DictConstants.DICT_TYPE_CONTRACT_STATUS,tContract.getStatus()));
+            FlwHisTask flwHisTask = flwHisTaskMapper.selectOne(new LambdaQueryWrapper<FlwHisTask>()
+                    .like(FlwHisTask::getVariable, tContract.getId())
+                    .last("LIMIT 1"));
+            if (Objects.nonNull(flwHisTask)){
+                tContract.setInstanceId(Objects.nonNull(flwHisTask.getInstanceId())?String.valueOf(flwHisTask.getInstanceId()):"");
+            }
         }
         pageInfo.setRecords(list);
         return pageInfo;
@@ -90,7 +96,7 @@
                     .like(FlwHisTask::getVariable, tContract.getId())
                     .last("LIMIT 1"));
             if (Objects.nonNull(flwHisTask)){
-                tContract.setInstanceId(flwHisTask.getInstanceId());
+                tContract.setInstanceId(Objects.nonNull(flwHisTask.getInstanceId())?String.valueOf(flwHisTask.getInstanceId()):"");
             }
         }
         pageInfo.setRecords(list);
diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TFlowManagementServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TFlowManagementServiceImpl.java
index 3bc07fe..9cf988d 100644
--- a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TFlowManagementServiceImpl.java
+++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TFlowManagementServiceImpl.java
@@ -33,7 +33,7 @@
                 .like(StringUtils.isNotEmpty(query.getPayer()),TFlowManagement::getPayer,query.getPayer())
                 .ge(StringUtils.isNotEmpty(query.getPayStartTime()),TFlowManagement::getPayTime,query.getPayStartTime())
                 .lt(StringUtils.isNotEmpty(query.getPayEndTime()),TFlowManagement::getPayTime,query.getPayEndTime())
-                .eq(null != query.getFlowStatus(),TFlowManagement::getFlowStatus,query.getFlowStatus())
+                .eq(null != query.getPayType(),TFlowManagement::getPayType,query.getPayType())
                 .orderByDesc(TFlowManagement::getCreateTime)
         ;
         return this.baseMapper.selectPage(pageInfo,queryWrapper);
diff --git a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TPayOrderServiceImpl.java b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TPayOrderServiceImpl.java
index 4d6ed9f..c8d77eb 100644
--- a/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TPayOrderServiceImpl.java
+++ b/ruoyi-system/src/main/java/com/ruoyi/system/service/impl/TPayOrderServiceImpl.java
@@ -69,12 +69,13 @@
             if (bill == null) {
                 throw new ServiceException("账单不存在");
             }
-            //计算欠费金额:租金+违约金-实收金额
-            long rent = caculateRentFee(bill);
+            //欠费金额转成单位分比较
+            long rent = bill.getOutstandingMoney().multiply(AmountConstant.b100)
+                    .setScale(2,RoundingMode.HALF_DOWN).longValue();
             if (rent==0){
                 throw new ServiceException("该账单已缴费");
             }
-            if (rent>dto.getAmount()){
+            if (rent<dto.getAmount()){
                 throw new ServiceException("支付金额超过了账单欠费金额");
             }
 
@@ -105,7 +106,11 @@
             orderBills.add(new TOrderBill(orderNo,bill.getId()));
             return bill;
         }).collect(Collectors.toList());
-        long sumRent = bills.stream().mapToLong((bill) -> caculateRentFee(bill)).sum();
+        //欠费金额转成单位分比较
+        long sumRent = bills.stream().mapToLong((bill)
+                -> bill.getOutstandingMoney().multiply(AmountConstant.b100)
+                .setScale(2,RoundingMode.HALF_DOWN).longValue())
+                .sum();
         if (sumRent==0){
             throw new ServiceException("账单已缴费");
         }
diff --git a/ruoyi-system/src/main/resources/mapper/system/TBankFlowMapper.xml b/ruoyi-system/src/main/resources/mapper/system/TBankFlowMapper.xml
index c72c8f5..8995310 100644
--- a/ruoyi-system/src/main/resources/mapper/system/TBankFlowMapper.xml
+++ b/ruoyi-system/src/main/resources/mapper/system/TBankFlowMapper.xml
@@ -37,9 +37,9 @@
     <!-- 统计总额和微信支付金额 -->
     <select id="getPaymentStats" resultMap="PaymentStatsResultMap">
         SELECT
-        SUM(flow_money) AS totalFlowMoney,
-        SUM(deduction_money) AS totalDeductionMoney,
-        SUM(remaining_money) AS totalRemainingMoney
+        ifnull(SUM(flow_money),0) AS totalFlowMoney,
+        ifnull(SUM(deduction_money),0) AS totalDeductionMoney,
+        ifnull(SUM(remaining_money),0) AS totalRemainingMoney
         FROM
         t_bank_flow
         <where>
diff --git a/ruoyi-system/src/main/resources/mapper/system/TBillMapper.xml b/ruoyi-system/src/main/resources/mapper/system/TBillMapper.xml
index 1e5bc53..5415d26 100644
--- a/ruoyi-system/src/main/resources/mapper/system/TBillMapper.xml
+++ b/ruoyi-system/src/main/resources/mapper/system/TBillMapper.xml
@@ -63,7 +63,7 @@
                 and t.id = #{query.userId}
             </if>
         </where>
-        order by b.payable_fees_time
+        order by b.payable_fees_time desc,b.create_time desc
     </select>
     <select id="getBillList" resultType="com.ruoyi.system.dto.TBillDto">
         SELECT
@@ -148,18 +148,18 @@
     </select>
 
     <select id="statisticsAllRent" resultType="java.math.BigDecimal">
-        SELECT sum(payable_fees_money) as amount FROM t_bill
+        SELECT ifnull(sum(payable_fees_money),0) as amount FROM t_bill
     </select>
 
     <select id="statisticsNoPay" resultType="java.math.BigDecimal">
-        SELECT sum(outstanding_money) as amount FROM t_bill where pay_fees_status!=3
+        SELECT ifnull(sum(outstanding_money),0) as amount FROM t_bill where pay_fees_status!=3
     </select>
 
     <select id="statisticsPayed" resultType="java.math.BigDecimal">
-        SELECT sum(pay_fees_money) as amount FROM t_bill
+        SELECT ifnull(sum(pay_fees_money),0) as amount FROM t_bill
     </select>
 
     <select id="statisticsOverdue" resultType="java.math.BigDecimal">
-        SELECT sum(outstanding_money) as amount FROM t_bill where pay_fees_status=4
+        SELECT ifnull(sum(outstanding_money),0) as amount FROM t_bill where pay_fees_status=4
     </select>
 </mapper>
diff --git a/ruoyi-system/src/main/resources/mapper/system/TFlowManagementMapper.xml b/ruoyi-system/src/main/resources/mapper/system/TFlowManagementMapper.xml
index f11b0fd..136ec5f 100644
--- a/ruoyi-system/src/main/resources/mapper/system/TFlowManagementMapper.xml
+++ b/ruoyi-system/src/main/resources/mapper/system/TFlowManagementMapper.xml
@@ -39,10 +39,10 @@
 
     <!-- 统计总额和微信支付金额 -->
     <select id="getPaymentStats" resultMap="PaymentStatsResultMap">
-        SELECT SUM(flow_money) AS total_amount,
-               SUM(CASE WHEN pay_type = 1 THEN flow_money ELSE 0 END ) AS wechat_amount,
-               SUM(CASE WHEN pay_type = 2 THEN flow_money ELSE 0 END ) AS alipay_amount,
-               SUM(CASE WHEN pay_type = 3 THEN flow_money ELSE 0 END ) AS offline_amount
+        SELECT ifnull(SUM(flow_money),0) AS total_amount,
+        ifnull(SUM(CASE WHEN pay_type = 1 THEN flow_money ELSE 0 END ) ,0)AS wechat_amount,
+        ifnull(SUM(CASE WHEN pay_type = 2 THEN flow_money ELSE 0 END ),0) AS alipay_amount,
+        ifnull(SUM(CASE WHEN pay_type = 3 THEN flow_money ELSE 0 END ),0) AS offline_amount
         FROM
             t_flow_management
         <where>

--
Gitblit v1.7.1