Blame view

service-fee/src/main/java/com/java110/fee/bill/CycleConvertOnceFeeImpl.java 14.9 KB
88e030b7   王彪总   init project
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
  package com.java110.fee.bill;
  
  import com.java110.core.factory.CommunitySettingFactory;
  import com.java110.core.factory.GenerateCodeFactory;
  import com.java110.core.log.LoggerFactory;
  import com.java110.core.smo.IComputeFeeSMO;
  import com.java110.dto.fee.FeeAttrDto;
  import com.java110.dto.fee.FeeConfigDto;
  import com.java110.dto.fee.FeeDto;
  import com.java110.dto.owner.OwnerDto;
  import com.java110.dto.payFee.PayFeeDetailRefreshFeeMonthDto;
  import com.java110.dto.payFee.PayFeeRuleDto;
  import com.java110.fee.dao.impl.PayFeeV1ServiceDaoImpl;
  import com.java110.intf.fee.*;
  import com.java110.po.fee.FeeAttrPo;
  import com.java110.po.fee.PayFeePo;
  import com.java110.po.payFee.PayFeeRulePo;
  import com.java110.po.payFee.PayFeeRuleBillPo;
  import com.java110.utils.util.*;
  import org.slf4j.Logger;
  import org.springframework.beans.factory.annotation.Autowired;
  import org.springframework.stereotype.Service;
  
  import java.util.ArrayList;
  import java.util.Calendar;
  import java.util.Date;
  import java.util.List;
  
  /**
   * 转换一次性费用
   */
  
  @Service
  public class CycleConvertOnceFeeImpl implements ICycleConvertOnceFee {
  
      private static Logger logger = LoggerFactory.getLogger(PayFeeV1ServiceDaoImpl.class);
  
  
      @Autowired
      private IFeeInnerServiceSMO feeInnerServiceSMOImpl;
  
      @Autowired
      private IPayFeeConfigV1InnerServiceSMO payFeeConfigV1InnerServiceSMOImpl;
  
      @Autowired
      private IPayFeeRuleV1InnerServiceSMO payFeeRuleV1InnerServiceSMOImpl;
  
      @Autowired
      private IPayFeeV1InnerServiceSMO payFeeV1InnerServiceSMOImpl;
  
      @Autowired
      private IFeeAttrInnerServiceSMO feeAttrInnerServiceSMOImpl;
  
      @Autowired
      private IPayFeeMonthInnerServiceSMO payFeeMonthInnerServiceSMOImpl;
  
      @Autowired
      private IComputeFeeSMO computeFeeSMOImpl;
  
      @Autowired
      private IPayFeeRuleBillV1InnerServiceSMO payFeeRuleBillV1InnerServiceSMOImpl;
  
      @Override
      public int convertPayFees(List<PayFeePo> payFeePos) {
  
          List<PayFeePo> tmpPayFeePos = new ArrayList<>();
          for (PayFeePo tmpPayFeePo : payFeePos) {
              //todo 一次性费用 直接跳过
              if (FeeDto.FEE_FLAG_ONCE.equals(tmpPayFeePo.getFeeFlag())) {
                  continue;
              }
  
              tmpPayFeePos.add(tmpPayFeePo);
          }
          if (tmpPayFeePos.isEmpty()) {
              return 0;
          }
  
          //todo 业务处理
  
          List<PayFeeRulePo> payFeeRulePos = new ArrayList<>();
  
          PayFeeRulePo tmpPayFeeRulePo = null;
          String curYearMonth = "";
          for (PayFeePo tmpPayFeePo : payFeePos) {
              tmpPayFeeRulePo = BeanConvertUtil.covertBean(tmpPayFeePo, PayFeeRulePo.class);
              tmpPayFeeRulePo.setRuleId(GenerateCodeFactory.getGeneratorId("11"));
              curYearMonth = DateUtil.getFormatTimeStringB(DateUtil.getDateFromStringB(tmpPayFeePo.getEndTime()));
              tmpPayFeeRulePo.setCurYearMonth(curYearMonth);
              payFeeRulePos.add(tmpPayFeeRulePo);
          }
  
          if (payFeeRulePos.isEmpty()) {
              return 0;
          }
  
          //todo 保存规则
          payFeeRuleV1InnerServiceSMOImpl.savePayFeeRules(payFeeRulePos);
  
          //todo 根据规则生成费用
          rulesGeneratePayFees(payFeeRulePos);
  
          return 0;
      }
  
      @Override
      public int convertPayFee(PayFeePo payFeePo) {
          List<PayFeePo> payFeePos = new ArrayList<>();
          payFeePos.add(payFeePo);
          return convertPayFees(payFeePos);
      }
  
      @Override
      public int covertCommunityPayFee(String communityId) {
          PayFeeRuleDto payFeeRuleDto = new PayFeeRuleDto();
          payFeeRuleDto.setCommunityId(communityId);
          List<PayFeeRuleDto> payFeeRuleDtos = payFeeRuleV1InnerServiceSMOImpl.queryPayFeeRules(payFeeRuleDto);
          if (payFeeRuleDtos == null || payFeeRuleDtos.isEmpty()) {
              return 0;
          }
  
          List<PayFeeRulePo> tmpPayFeeRulePos = BeanConvertUtil.covertBeanList(payFeeRuleDtos, PayFeeRulePo.class);
          return rulesGeneratePayFees(tmpPayFeeRulePos);
      }
  
      @Override
      public int covertRuleIdsPayFee(List<String> ruleIds) {
          PayFeeRuleDto payFeeRuleDto = new PayFeeRuleDto();
          payFeeRuleDto.setRuleIds(ruleIds.toArray(new String[ruleIds.size()]));
          List<PayFeeRuleDto> payFeeRuleDtos = payFeeRuleV1InnerServiceSMOImpl.queryPayFeeRules(payFeeRuleDto);
          if (payFeeRuleDtos == null || payFeeRuleDtos.isEmpty()) {
              return 0;
          }
  
          List<PayFeeRulePo> tmpPayFeeRulePos = BeanConvertUtil.covertBeanList(payFeeRuleDtos, PayFeeRulePo.class);
          return rulesGeneratePayFees(tmpPayFeeRulePos);
      }
  
      @Override
      public int rulesGeneratePayFees(List<PayFeeRulePo> payFeeRulePos) {
  
          if (ListUtil.isNull(payFeeRulePos)) {
              return 0;
          }
  
          for (PayFeeRulePo tmpPayFeeRulePo : payFeeRulePos) {
              try {
                  //todo 单个费用处理
                  ruleGeneratePayFee(tmpPayFeeRulePo);
              } catch (Exception e) {
                  logger.error("处理异常 ruleId" + tmpPayFeeRulePo.getRuleId(), e);
              }
          }
          return payFeeRulePos.size();
      }
  
      /**
       * 规则生成费用
       *
       * @param tmpPayFeeRulePo
       */
      public int ruleGeneratePayFee(PayFeeRulePo tmpPayFeeRulePo) {
          int ruleCycle = 1;
          String value = CommunitySettingFactory.getValue(tmpPayFeeRulePo.getCommunityId(), "PAY_FEE_MONTH_CYCLE");
          if (StringUtil.isNumber(value)) {
              ruleCycle = Integer.parseInt(value);
          }
  
          //todo 查询费用项信息
          FeeConfigDto feeConfigDto = new FeeConfigDto();
          feeConfigDto.setConfigId(tmpPayFeeRulePo.getConfigId());
          feeConfigDto.setCommunityId(tmpPayFeeRulePo.getCommunityId());
          List<FeeConfigDto> feeConfigDtos = payFeeConfigV1InnerServiceSMOImpl.queryPayFeeConfigs(feeConfigDto);
  
          Assert.listOnlyOne(feeConfigDtos, "费用项不存在,configId=" + tmpPayFeeRulePo.getConfigId());
  
          //todo 计算目标 结束时间
          Date targetEndTime = computeTargetEndTime(tmpPayFeeRulePo, feeConfigDtos.get(0));
  
          //todo 创建 pay_fee 和 attrs 数据
          List<PayFeeRuleBillPo> payFeeRuleBillPos = new ArrayList<>();
          List<PayFeePo> tmpPayFeePos = new ArrayList<>();
          List<FeeAttrPo> tmpFeeAttrPos = new ArrayList<>();
  
          FeeDto feeDto = new FeeDto();
          feeDto.setPayerObjType(tmpPayFeeRulePo.getPayerObjType());
          feeDto.setCommunityId(tmpPayFeeRulePo.getCommunityId());
          feeDto.setPayerObjId(tmpPayFeeRulePo.getPayerObjId());
          OwnerDto ownerDto = computeFeeSMOImpl.getFeeOwnerDto(feeDto);
          String payerObjName = computeFeeSMOImpl.getFeeObjName(feeDto);
  
          Date startTime = DateUtil.getDateFromStringB(tmpPayFeeRulePo.getCurYearMonth());
          Date endTime = null;
  
          //todo 没到时间
          if (startTime.getTime() >= targetEndTime.getTime()) {
              return 0;
          }
  
          do {
              endTime = DateUtil.getNextMonthFirstDate(startTime, ruleCycle);
              if (endTime.getTime() >= targetEndTime.getTime()) {
                  endTime = targetEndTime;
              }
              //todo 生成 费用
              doGeneratorPayFee(tmpPayFeeRulePo, startTime, endTime, feeConfigDtos.get(0), tmpPayFeePos, tmpFeeAttrPos, payFeeRuleBillPos, ownerDto, payerObjName,ruleCycle);
  
              startTime = endTime;
  
          }
          while (endTime.getTime() < targetEndTime.getTime());
          int saveFlag = 0;
          if (!tmpPayFeePos.isEmpty()) {
              saveFlag = saveFeeAndAttrs(tmpPayFeePos, tmpFeeAttrPos, payFeeRuleBillPos);
          }
  
          //todo 修改pay_fee_rule 的CurYearMonth
  
          PayFeeRulePo payFeeRulePo = new PayFeeRulePo();
          payFeeRulePo.setRuleId(tmpPayFeeRulePo.getRuleId());
          payFeeRulePo.setCurYearMonth(DateUtil.getFormatTimeStringB(endTime));
          payFeeRulePo.setCommunityId(tmpPayFeeRulePo.getCommunityId());
          payFeeRuleV1InnerServiceSMOImpl.updatePayFeeRule(payFeeRulePo);
  
          return saveFlag;
      }
  
  
      private void doGeneratorPayFee(PayFeeRulePo tmpPayFeeRulePo, Date startTime, Date endTime, FeeConfigDto feeConfigDto,
                                     List<PayFeePo> tmpPayFeePos, List<FeeAttrPo> tmpFeeAttrPos,
                                     List<PayFeeRuleBillPo> payFeeRuleBillPos,
                                     OwnerDto ownerDto,
                                     String payerObjName,
                                     int ruleCycle) {
  
          PayFeePo payFeePo = BeanConvertUtil.covertBean(tmpPayFeeRulePo, PayFeePo.class);
          payFeePo.setFeeId(GenerateCodeFactory.getGeneratorId(GenerateCodeFactory.CODE_PREFIX_feeId));
          payFeePo.setFeeFlag(FeeDto.FEE_FLAG_ONCE);
          payFeePo.setStartTime(DateUtil.getFormatTimeStringB(startTime));
          payFeePo.setEndTime(DateUtil.getFormatTimeStringB(startTime));
          payFeePo.setRuleId(tmpPayFeeRulePo.getRuleId());
  
          tmpPayFeePos.add(payFeePo);
  
          //todo 生成ruleBill 数据
          PayFeeRuleBillPo payFeeRuleBillPo = new PayFeeRuleBillPo();
          payFeeRuleBillPo.setFeeId(payFeePo.getFeeId());
          payFeeRuleBillPo.setBillId(GenerateCodeFactory.getGeneratorId("13"));
          payFeeRuleBillPo.setBillName(feeConfigDto.getFeeName());
          payFeeRuleBillPo.setRuleId(tmpPayFeeRulePo.getRuleId());
          payFeeRuleBillPo.setConfigId(tmpPayFeeRulePo.getConfigId());
          payFeeRuleBillPo.setBatchId(payFeePo.getBatchId());
          payFeeRuleBillPo.setCurYearMonth(DateUtil.getFormatTimeStringB(startTime));
          payFeeRuleBillPo.setCommunityId(payFeePo.getCommunityId());
          payFeeRuleBillPo.setMonthCycle(ruleCycle+"");
          payFeeRuleBillPos.add(payFeeRuleBillPo);
  
          //todo 这里时间少一天,显示,因为是周期性转一次性 所以 这里为了显示合理 结束时间少一天处理
          Calendar endTimeCal = Calendar.getInstance();
          endTimeCal.setTime(endTime);
          endTimeCal.add(Calendar.DAY_OF_MONTH, -1);
          tmpFeeAttrPos.add(addFeeAttr(payFeePo, FeeAttrDto.SPEC_CD_ONCE_FEE_DEADLINE_TIME, DateUtil.getFormatTimeStringB(endTimeCal.getTime())));
  
          if (ownerDto != null) {
              tmpFeeAttrPos.add(addFeeAttr(payFeePo, FeeAttrDto.SPEC_CD_OWNER_ID, ownerDto.getOwnerId()));
              tmpFeeAttrPos.add(addFeeAttr(payFeePo, FeeAttrDto.SPEC_CD_OWNER_LINK, ownerDto.getLink()));
              tmpFeeAttrPos.add(addFeeAttr(payFeePo, FeeAttrDto.SPEC_CD_OWNER_NAME, ownerDto.getName()));
          }
  
          //todo 付费对象名称
          tmpFeeAttrPos.add(addFeeAttr(payFeePo, FeeAttrDto.SPEC_CD_PAY_OBJECT_NAME, payerObjName));
  
  
      }
  
      private FeeAttrPo addFeeAttr(PayFeePo payFeePo, String specCd, String value) {
          FeeAttrPo feeAttrPo = new FeeAttrPo();
          feeAttrPo.setCommunityId(payFeePo.getCommunityId());
          feeAttrPo.setSpecCd(specCd);
          feeAttrPo.setValue(value);
          feeAttrPo.setFeeId(payFeePo.getFeeId());
          feeAttrPo.setAttrId(GenerateCodeFactory.getGeneratorId(GenerateCodeFactory.CODE_PREFIX_attrId, true));
          return feeAttrPo;
      }
  
  
      private int saveFeeAndAttrs(List<PayFeePo> feePos, List<FeeAttrPo> feeAttrsPos, List<PayFeeRuleBillPo> payFeeRuleBillPos) {
          if (feePos == null || feePos.isEmpty()) {
              return 1;
          }
          int flag = feeInnerServiceSMOImpl.saveFee(feePos);
          if (flag < 1) {
              return flag;
          }
  
          if (!payFeeRuleBillPos.isEmpty()) {
              payFeeRuleBillV1InnerServiceSMOImpl.savePayFeeRuleBills(payFeeRuleBillPos);
          }
  
          flag = feeAttrInnerServiceSMOImpl.saveFeeAttrs(feeAttrsPos);
  
          // todo 这里异步的方式计算 月数据 和欠费数据
          List<String> feeIds = new ArrayList<>();
          for (PayFeePo feePo : feePos) {
              feeIds.add(feePo.getFeeId());
          }
  
          PayFeeDetailRefreshFeeMonthDto payFeeDetailRefreshFeeMonthDto = new PayFeeDetailRefreshFeeMonthDto();
          payFeeDetailRefreshFeeMonthDto.setCommunityId(feePos.get(0).getCommunityId());
          payFeeDetailRefreshFeeMonthDto.setFeeIds(feeIds);
  
          payFeeMonthInnerServiceSMOImpl.doGeneratorFeeMonths(payFeeDetailRefreshFeeMonthDto);
  
          payFeeMonthInnerServiceSMOImpl.doGeneratorOweFees(payFeeDetailRefreshFeeMonthDto);
          return flag;
      }
  
      /**
       * 计算目标结束时间
       *
       * @param tmpPayFeeRulePo
       * @param feeConfigDto
       * @return
       */
      private Date computeTargetEndTime(PayFeeRulePo tmpPayFeeRulePo, FeeConfigDto feeConfigDto) {
  
          Date targetEndDate = null;
          //todo 判断当前费用是否已结束
          if (FeeDto.STATE_FINISH.equals(tmpPayFeeRulePo.getState())) {
              targetEndDate = DateUtil.getDateFromStringB(tmpPayFeeRulePo.getEndTime());
              return targetEndDate;
          }
  
          Calendar preEndTimeCal = Calendar.getInstance();
          preEndTimeCal.setTime(DateUtil.getDateFromStringB(tmpPayFeeRulePo.getEndTime()));
          if (StringUtil.isNumber(feeConfigDto.getPrepaymentPeriod())) {
              preEndTimeCal.add(Calendar.DAY_OF_MONTH, Integer.parseInt(feeConfigDto.getPrepaymentPeriod()) * -1);
          }
          Date preEndTime = preEndTimeCal.getTime();
          Date maxEndTime = DateUtil.getDateFromStringB(tmpPayFeeRulePo.getMaxTime());
  
          Date billEndTime = DateUtil.getCurrentDate();
          //建账时间
          Date startDate = DateUtil.getDateFromStringB(tmpPayFeeRulePo.getStartTime());
          //计费起始时间
          Date endDate = DateUtil.getDateFromStringB(tmpPayFeeRulePo.getEndTime());
          //缴费周期
          long paymentCycle = Long.parseLong(feeConfigDto.getPaymentCycle());
          // 当前时间 - 开始时间  = 月份
          double mulMonth = 0.0;
          mulMonth = DateUtil.dayCompare(startDate, billEndTime);
  
          // 月份/ 周期 = 轮数(向上取整)
          double round = 0.0;
          if ("1200".equals(feeConfigDto.getPaymentCd())) { // 1200预付费
              round = Math.floor(mulMonth / paymentCycle) + 1;
          } else { //2100后付费
              round = Math.floor(mulMonth / paymentCycle);
          }
          // 轮数 * 周期 * 30 + 开始时间 = 目标 到期时间
          targetEndDate = DateUtil.getTargetEndTime(round * paymentCycle, startDate);//目标结束时间
  
          //todo 如果 到了 预付期 产生下个周期的费用
          if (DateUtil.getFormatTimeStringB(targetEndDate).equals(DateUtil.getFormatTimeStringB(endDate))
                  && DateUtil.getCurrentDate().getTime() > preEndTime.getTime()
          ) {
              targetEndDate = DateUtil.getTargetEndTime((round + 1) * paymentCycle, startDate);//目标结束时间
          }
  
  
          //todo 费用项的结束时间<缴费的结束时间  费用快结束了   取费用项的结束时间
          if (maxEndTime.getTime() < targetEndDate.getTime()) {
              targetEndDate = maxEndTime;
          }
  
          if (DateUtil.getDateFromStringB(tmpPayFeeRulePo.getEndTime()).getTime() > targetEndDate.getTime()) {
              targetEndDate = DateUtil.getDateFromStringB(tmpPayFeeRulePo.getEndTime());
          }
  
          return targetEndDate;
      }
  }