123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307 |
- package com.simuwang.manage.task;
- import cn.hutool.core.date.DateUtil;
- import com.simuwang.base.common.enums.DeletionType;
- import com.simuwang.base.common.enums.DistributeType;
- import com.simuwang.base.common.enums.Frequency;
- import com.simuwang.base.common.util.DateUtils;
- import com.simuwang.base.common.util.StringUtil;
- import com.simuwang.base.mapper.*;
- import com.simuwang.base.pojo.dos.*;
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.scheduling.annotation.EnableScheduling;
- import org.springframework.scheduling.annotation.Scheduled;
- import org.springframework.stereotype.Component;
- import java.math.BigDecimal;
- import java.util.*;
- import java.util.stream.Collectors;
- /**
- * FileName: FundDeletionTask
- * Author: chenjianhua
- * Date: 2024/9/18 22:43
- * Description: ${DESCRIPTION}
- */
- @EnableScheduling
- @Component
- public class FundDeletionTask {
- @Autowired
- private NavMapper navMapper;
- @Autowired
- private AssetMapper assetMapper;
- @Autowired
- private FundInfoMapper fundInfoMapper;
- @Autowired
- private FundReportFrequencyMapper fundReportFrequencyMapper;
- @Autowired
- private TradeDateMapper tradeDateMapper;
- @Autowired
- private DeletionInfoMapper deletionInfoMapper;
- @Autowired
- private DistributionMapper distributionMapper;
- // @Scheduled(cron = "0 0 5,12,18 * * ?")
- public void computeDeletion(){
- List<String> fundIdList = navMapper.getAllFundId();
- for(String fundId : fundIdList){
- String liquidateDate = fundInfoMapper.getLiquidateDateByFundId(fundId);
- if(StringUtil.isNotEmpty(liquidateDate)){
- continue;
- }
- String inceptionDate = fundInfoMapper.getInceptionDateByFundId(fundId);
- if(StringUtil.isEmpty(inceptionDate)){
- continue;
- }
- String today = DateUtils.getAroundToday(0);
- //获取基金对应的净值报送频率
- FundReportFrequencyDO fundReportFrequencyDO = fundReportFrequencyMapper.getFrequencyByFundId(fundId);
- if(StringUtil.isNull(fundReportFrequencyDO)){
- continue;
- }
- List<NavDO> navDOList = navMapper.selectNavByFundId(fundId);
- List<AssetDO> assetDOList = assetMapper.selectAssetByFundId(fundId);
- //查询成立日到今天为止的交易日集合
- List<TradeDateDO> tradeDateDOList = tradeDateMapper.selectTradeDate(inceptionDate,today);
- //延迟天数为3天
- tradeDateDOList = tradeDateDOList.subList(0,tradeDateDOList.size()-3);
- navDeletion(fundId,navDOList,tradeDateDOList,fundReportFrequencyDO);
- assetDeletion(fundId,assetDOList,tradeDateDOList,fundReportFrequencyDO);
- distributionDeletion(fundId,navDOList);
- }
- }
- private void distributionDeletion(String fundId, List<NavDO> navDOList) {
- if(navDOList.size() < 1){
- return;
- }
- //查询是否存在拆分
- List<DistributionDO> distributionDOS = distributionMapper.getDistributionByFundId(fundId, DistributeType.DIVIDENDS_SPLIT);
- if(distributionDOS.size() > 0){
- //存在拆分,不做分红缺失计算,同时把以往的数据添加备注
- deletionInfoMapper.updateRemark(fundId,DeletionType.ASSET_DELETION.getCode(),null,DeletionType.EXIST_SPLIT.getInfo(),null);
- return;
- }
- BigDecimal threshold = new BigDecimal(0.0035);
- BigDecimal preDifference = new BigDecimal(0);
- for(int navIdx=0;navIdx < navDOList.size() ;navIdx++){
- NavDO navDO = navDOList.get(navIdx);
- //获取当前净值日期下的分红总和
- BigDecimal sumDistribute = distributionMapper.getSumDistributeByFundId(navDO.getFundId(),DateUtils.format(navDO.getPriceDate(),DateUtils.YYYY_MM_DD));
- BigDecimal nav = navDO.getNav();
- BigDecimal cumulativeNavWithdrawal = navDO.getCumulativeNavWithdrawal();
- if(sumDistribute == null){
- sumDistribute = new BigDecimal(0);
- }
- //不存在分红的时候,判断当前的差值是否符合要求
- BigDecimal difference = cumulativeNavWithdrawal.subtract(nav.add(sumDistribute));
- if(difference.compareTo(threshold) > 0){
- //存在缺失
- String tradeDate = null;
- if(navIdx == 0 || navDOList.size() ==1){
- //有且仅有一条数据
- tradeDate = DateUtils.format(navDO.getPriceDate(),DateUtils.YYYY_MM_DD)+"~"+DateUtils.format(navDO.getPriceDate(),DateUtils.YYYY_MM_DD);
- }else{
- //判断前一条的差值与当前是否一致,一致就不处理
- if(difference.compareTo(preDifference) == 0){
- continue;
- }
- tradeDate = DateUtils.format(navDOList.get(navIdx-1).getPriceDate(),DateUtils.YYYY_MM_DD)+"~"+DateUtils.format(navDO.getPriceDate(),DateUtils.YYYY_MM_DD);
- }
- saveDeletionInfoDO(fundId,tradeDate,DeletionType.DISTRIBUTION_DELETION.getCode());
- }
- preDifference = difference;
- }
- }
- private void assetDeletion(String fundId, List<AssetDO> assetDOList, List<TradeDateDO> tradeDateDOList, FundReportFrequencyDO fundReportFrequencyDO) {
- if(Frequency.DAY == Frequency.getFrequencyByCode(fundReportFrequencyDO.getAssetFrequency())){
- Map<String,List<AssetDO>> navListMap = assetDOList.stream().collect(Collectors.groupingBy(e -> DateUtils.format(e.getPriceDate(),DateUtils.YYYY_MM_DD)));
- Map<String,List<TradeDateDO>> tradeListMap = tradeDateDOList.stream().collect(Collectors.groupingBy(e -> DateUtils.format(e.getTradeDate(),DateUtils.YYYY_MM_DD)));
- for(String tradeDate : tradeListMap.keySet()){
- if(navListMap.containsKey(tradeDate)){
- deletionInfoMapper.updateRemark(fundId,DeletionType.ASSET_DELETION.getCode(),tradeDate,DeletionType.NO_DELETION.getInfo(),null);
- continue;
- }
- //写入缺失信息表
- saveDeletionInfoDO(fundId,tradeDate,DeletionType.ASSET_DELETION.getCode());
- }
- }
- if(Frequency.WEEK == Frequency.getFrequencyByCode(fundReportFrequencyDO.getAssetFrequency())){
- Map<String,List<AssetDO>> navListMap = assetDOList.stream().collect(Collectors.groupingBy(e -> DateUtils.format(e.getPriceDate(),DateUtils.YYYY_MM_DD)));
- TreeMap<Integer,List<AssetDO>> weekNavListMap = new TreeMap<>();
- //按周数整合
- for(String priceDate : navListMap.keySet()){
- Integer weekOfYear = DateUtil.weekOfYear(DateUtils.parse(priceDate,DateUtils.YYYY_MM_DD));
- if(weekNavListMap.containsKey(weekOfYear)){
- List<AssetDO> assetDOS = weekNavListMap.get(weekOfYear);
- assetDOS.addAll(navListMap.get(priceDate));
- weekNavListMap.put(weekOfYear,assetDOS);
- }else{
- List<AssetDO> navDOS = new ArrayList<>();
- navDOS.addAll(navListMap.get(priceDate));
- weekNavListMap.put(weekOfYear,navDOS);
- }
- }
- Map<Integer,List<TradeDateDO>> tradeListMap = tradeDateDOList.stream().collect(Collectors.groupingBy(e -> e.getWeekOfYear()));
- for(Integer weekOfYear : tradeListMap.keySet()){
- if(weekNavListMap.containsKey(weekOfYear)){
- List<AssetDO> assetDOS = weekNavListMap.get(weekOfYear);
- String tradeDate = DateUtils.format(assetDOS.get(assetDOS.size()-1).getPriceDate(),DateUtils.YYYY_MM_DD);
- deletionInfoMapper.updateRemark(fundId,DeletionType.ASSET_DELETION.getCode(),tradeDate,DeletionType.NO_DELETION.getInfo(),null);
- continue;
- }
- //不包含的话,默认取每周的最后一个交易日作为周净值日期
- List<TradeDateDO> tradeDateDOS = tradeListMap.get(weekOfYear);
- String tradeDate = null;
- if(tradeDateDOS.size() > 0){
- tradeDate = DateUtils.format(tradeDateDOS.get(tradeDateDOS.size()-1).getTradeDate(),DateUtils.YYYY_MM_DD);
- }
- //写入缺失信息表
- saveDeletionInfoDO(fundId,tradeDate,DeletionType.NAV_DELETION.getCode());
- }
- }
- if(Frequency.MONTH == Frequency.getFrequencyByCode(fundReportFrequencyDO.getAssetFrequency())){
- Map<String,List<AssetDO>> navListMap = assetDOList.stream().collect(Collectors.groupingBy(e -> DateUtils.format(e.getPriceDate(),DateUtils.YYYY_MM_DD)));
- TreeMap<String,List<AssetDO>> monthNavListMap = new TreeMap<>();
- //按周数整合
- for(String priceDate : navListMap.keySet()){
- String yearMonth = priceDate.substring(0,7);
- if(monthNavListMap.containsKey(yearMonth)){
- List<AssetDO> navDOS = monthNavListMap.get(yearMonth);
- navDOS.addAll(navListMap.get(priceDate));
- monthNavListMap.put(yearMonth,navDOS);
- }else{
- List<AssetDO> navDOS = new ArrayList<>();
- navDOS.addAll(navListMap.get(priceDate));
- monthNavListMap.put(yearMonth,navDOS);
- }
- }
- Map<String,List<TradeDateDO>> tradeListMap = tradeDateDOList.stream().collect(Collectors.groupingBy(e -> e.getYearmonth()));
- for(String yearMonth : tradeListMap.keySet()){
- if(monthNavListMap.containsKey(yearMonth)){
- List<AssetDO> assetDOS = monthNavListMap.get(yearMonth);
- String tradeDate = DateUtils.format(assetDOS.get(assetDOS.size()-1).getPriceDate(),DateUtils.YYYY_MM_DD);
- deletionInfoMapper.updateRemark(fundId,DeletionType.ASSET_DELETION.getCode(),tradeDate,DeletionType.NO_DELETION.getInfo(),null);
- continue;
- }
- //不包含的话,默认取每周的最后一个交易日作为周净值日期
- List<TradeDateDO> tradeDateDOS = tradeListMap.get(yearMonth);
- String tradeDate = null;
- if(tradeDateDOS.size() > 0){
- tradeDate = DateUtils.format(tradeDateDOS.get(tradeDateDOS.size()-1).getTradeDate(),DateUtils.YYYY_MM_DD);
- }
- //写入缺失信息表
- saveDeletionInfoDO(fundId,tradeDate,DeletionType.ASSET_DELETION.getCode());
- }
- }
- }
- private void navDeletion(String fundId,List<NavDO> navDOList, List<TradeDateDO> tradeDateDOList,FundReportFrequencyDO fundReportFrequencyDO) {
- //只处理日月季频率
- if(Frequency.DAY == Frequency.getFrequencyByCode(fundReportFrequencyDO.getNavFrequency())){
- Map<String,List<NavDO>> navListMap = navDOList.stream().collect(Collectors.groupingBy(e -> DateUtils.format(e.getPriceDate(),DateUtils.YYYY_MM_DD)));
- Map<String,List<TradeDateDO>> tradeListMap = tradeDateDOList.stream().collect(Collectors.groupingBy(e -> DateUtils.format(e.getTradeDate(),DateUtils.YYYY_MM_DD)));
- for(String tradeDate : tradeListMap.keySet()){
- if(navListMap.containsKey(tradeDate)){
- deletionInfoMapper.updateRemark(fundId,DeletionType.NAV_DELETION.getCode(),tradeDate,DeletionType.NO_DELETION.getInfo(),null);
- continue;
- }
- //写入缺失信息表
- saveDeletionInfoDO(fundId,tradeDate,DeletionType.NAV_DELETION.getCode());
- }
- }
- if(Frequency.WEEK == Frequency.getFrequencyByCode(fundReportFrequencyDO.getNavFrequency())){
- Map<String,List<NavDO>> navListMap = navDOList.stream().collect(Collectors.groupingBy(e -> DateUtils.format(e.getPriceDate(),DateUtils.YYYY_MM_DD)));
- TreeMap<Integer,List<NavDO>> weekNavListMap = new TreeMap<>();
- //按周数整合
- for(String priceDate : navListMap.keySet()){
- Integer weekOfYear = DateUtil.weekOfYear(DateUtils.parse(priceDate,DateUtils.YYYY_MM_DD));
- if(weekNavListMap.containsKey(weekOfYear)){
- List<NavDO> navDOS = weekNavListMap.get(weekOfYear);
- navDOS.addAll(navListMap.get(priceDate));
- weekNavListMap.put(weekOfYear,navDOS);
- }else{
- List<NavDO> navDOS = new ArrayList<>();
- navDOS.addAll(navListMap.get(priceDate));
- weekNavListMap.put(weekOfYear,navDOS);
- }
- }
- Map<Integer,List<TradeDateDO>> tradeListMap = tradeDateDOList.stream().collect(Collectors.groupingBy(e -> e.getWeekOfYear()));
- for(Integer weekOfYear : tradeListMap.keySet()){
- if(weekNavListMap.containsKey(weekOfYear)){
- List<NavDO> navDOS = weekNavListMap.get(weekOfYear);
- String tradeDate = DateUtils.format(navDOS.get(navDOS.size()-1).getPriceDate(),DateUtils.YYYY_MM_DD);
- deletionInfoMapper.updateRemark(fundId,DeletionType.NAV_DELETION.getCode(),tradeDate,DeletionType.NO_DELETION.getInfo(),null);
- continue;
- }
- //不包含的话,默认取每周的最后一个交易日作为周净值日期
- List<TradeDateDO> tradeDateDOS = tradeListMap.get(weekOfYear);
- String tradeDate = null;
- if(tradeDateDOS.size() > 0){
- tradeDate = DateUtils.format(tradeDateDOS.get(tradeDateDOS.size()-1).getTradeDate(),DateUtils.YYYY_MM_DD);
- }
- //写入缺失信息表
- saveDeletionInfoDO(fundId,tradeDate,DeletionType.NAV_DELETION.getCode());
- }
- }
- if(Frequency.MONTH == Frequency.getFrequencyByCode(fundReportFrequencyDO.getNavFrequency())){
- Map<String,List<NavDO>> navListMap = navDOList.stream().collect(Collectors.groupingBy(e -> DateUtils.format(e.getPriceDate(),DateUtils.YYYY_MM_DD)));
- TreeMap<String,List<NavDO>> monthNavListMap = new TreeMap<>();
- //按周数整合
- for(String priceDate : navListMap.keySet()){
- String yearMonth = priceDate.substring(0,7);
- if(monthNavListMap.containsKey(yearMonth)){
- List<NavDO> navDOS = monthNavListMap.get(yearMonth);
- navDOS.addAll(navListMap.get(priceDate));
- monthNavListMap.put(yearMonth,navDOS);
- }else{
- List<NavDO> navDOS = new ArrayList<>();
- navDOS.addAll(navListMap.get(priceDate));
- monthNavListMap.put(yearMonth,navDOS);
- }
- }
- Map<String,List<TradeDateDO>> tradeListMap = tradeDateDOList.stream().collect(Collectors.groupingBy(e -> e.getYearmonth()));
- for(String yearMonth : tradeListMap.keySet()){
- if(monthNavListMap.containsKey(yearMonth)){
- List<NavDO> navDOS = monthNavListMap.get(yearMonth);
- String tradeDate = DateUtils.format(navDOS.get(navDOS.size()-1).getPriceDate(),DateUtils.YYYY_MM_DD);
- deletionInfoMapper.updateRemark(fundId,DeletionType.NAV_DELETION.getCode(),tradeDate,DeletionType.NO_DELETION.getInfo(),null);
- continue;
- }
- //不包含的话,默认取每周的最后一个交易日作为周净值日期
- List<TradeDateDO> tradeDateDOS = tradeListMap.get(yearMonth);
- String tradeDate = null;
- if(tradeDateDOS.size() > 0){
- tradeDate = DateUtils.format(tradeDateDOS.get(tradeDateDOS.size()-1).getTradeDate(),DateUtils.YYYY_MM_DD);
- }
- //写入缺失信息表
- saveDeletionInfoDO(fundId,tradeDate,DeletionType.NAV_DELETION.getCode());
- }
- }
- }
- private void saveDeletionInfoDO(String fundId, String tradeDate, Integer code) {
- if(tradeDate == null){
- return;
- }
- DeletionInfoDO deletionInfoDO = new DeletionInfoDO();
- deletionInfoDO.setFundId(fundId);
- deletionInfoDO.setDeletionDate(tradeDate);
- deletionInfoDO.setDeletionType(code);
- DeletionInfoDO oldDeletionDO = deletionInfoMapper.getDeletionInfoDO(deletionInfoDO);
- if(StringUtil.isNull(oldDeletionDO)){
- deletionInfoDO.setIsvalid(1);
- deletionInfoDO.setIsSend(0);
- deletionInfoDO.setUpdateTime(DateUtils.getNowDate());
- deletionInfoDO.setCreateTime(DateUtils.getNowDate());
- deletionInfoMapper.saveDeletionInfoDO(deletionInfoDO);
- }
- }
- }
|