From 6c48164195944bbe12f0eca7728fba5c2ec277a9 Mon Sep 17 00:00:00 2001 From: "rong.liu" Date: Mon, 22 Feb 2021 11:44:12 +0800 Subject: [PATCH] =?UTF-8?q?=E6=9E=84=E5=BB=BA=E9=A1=B9=E7=9B=AE?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 33 + .mvn/wrapper/MavenWrapperDownloader.java | 118 +++ .mvn/wrapper/maven-wrapper.jar | Bin 0 -> 50710 bytes .mvn/wrapper/maven-wrapper.properties | 2 + README.md | 3 + mvnw | 310 ++++++ mvnw.cmd | 182 ++++ pom.xml | 236 +++++ .../java/com/msdw/tms/TmsApplication.java | 20 + .../java/com/msdw/tms/aop/AuthCustomize.java | 28 + src/main/java/com/msdw/tms/common/Test.java | 21 + .../tms/common/exception/CustomException.java | 21 + .../tms/common/exception/ExceptionCast.java | 12 + .../tms/common/exception/ExceptionCatch.java | 53 + .../tms/common/exception/RRException.java | 61 ++ .../msdw/tms/common/utils/AliyunOssUtil.java | 315 ++++++ .../com/msdw/tms/common/utils/BasePage.java | 19 + .../com/msdw/tms/common/utils/Constant.java | 489 +++++++++ .../msdw/tms/common/utils/DownloadUtils.java | 19 + .../msdw/tms/common/utils/FilesResult.java | 15 + .../tms/common/utils/MoneyConversionUtil.java | 27 + .../com/msdw/tms/common/utils/PageUtil.java | 22 + .../com/msdw/tms/common/utils/PageUtils.java | 112 ++ .../java/com/msdw/tms/common/utils/Query.java | 94 ++ .../java/com/msdw/tms/common/utils/R.java | 63 ++ .../msdw/tms/common/utils/RedisConfig.java | 41 + .../com/msdw/tms/common/utils/RestUtil.java | 186 ++++ .../com/msdw/tms/common/utils/TokenUtil.java | 84 ++ .../msdw/tms/common/utils/date/DateUtils.java | 99 ++ .../msdw/tms/common/utils/photo/photoUrl.java | 15 + .../tms/common/utils/poi/EasyExcelUtil.java | 28 + .../tms/common/utils/poi/ExcelAttribute.java | 25 + .../tms/common/utils/poi/ExcelExportUtil.java | 84 ++ .../tms/common/utils/poi/ExcelImportUtil.java | 123 +++ .../com/msdw/tms/common/utils/py/PyUtil.java | 211 ++++ .../utils/trading_rules/TradingDay.java | 167 +++ .../common/utils/trading_rules/algorithm.java | 434 ++++++++ .../com/msdw/tms/common/xss/HTMLFilter.java | 530 ++++++++++ .../com/msdw/tms/common/xss/SQLFilter.java | 49 + .../com/msdw/tms/config/AliyunOssConfig.java | 59 ++ .../java/com/msdw/tms/config/CorsConfig.java | 19 + .../com/msdw/tms/config/MyThreadConfig.java | 25 + .../msdw/tms/config/MybatisPlusConfig.java | 18 + .../java/com/msdw/tms/config/RedisConfig.java | 22 + .../com/msdw/tms/config/SwaggerConfig.java | 35 + .../config/ThreadPoolConfigProperties.java | 14 + .../com/msdw/tms/config/WebConfigurer.java | 37 + .../CdBankStatementsController.java | 324 ++++++ .../tms/controller/CdCacheController.java | 84 ++ .../controller/CdExperimentalController.java | 197 ++++ .../CdFuturesConfigureController.java | 189 ++++ .../tms/controller/CdHedgingController.java | 193 ++++ .../tms/controller/CdPointController.java | 88 ++ .../CdTransferWarehouseRecordController.java | 63 ++ .../controller/CdUserAssetsController.java | 87 ++ .../controller/QhscGangjiaosuoController.java | 209 ++++ .../WhscRenminbipinzhongController.java | 197 ++++ .../com/msdw/tms/dao/CdBankStatementsDao.java | 16 + .../java/com/msdw/tms/dao/CdCacheDao.java | 15 + .../com/msdw/tms/dao/CdExperimentalDao.java | 15 + .../msdw/tms/dao/CdFuturesConfigureDao.java | 15 + .../com/msdw/tms/dao/CdPointChildrenDao.java | 15 + .../java/com/msdw/tms/dao/CdPointDao.java | 15 + .../tms/dao/CdTransferWarehouseRecordDao.java | 15 + .../com/msdw/tms/dao/CdUserAssetsDao.java | 15 + .../msdw/tms/dao/CdUserOptionAccountDao.java | 15 + .../com/msdw/tms/dao/QhscGangjiaosuoDao.java | 28 + .../msdw/tms/dao/WhscRenminbipinzhongDao.java | 25 + .../tms/entity/CdBankStatementsEntity.java | 57 ++ .../com/msdw/tms/entity/CdCacheEntity.java | 62 ++ .../msdw/tms/entity/CdExperimentalEntity.java | 74 ++ .../tms/entity/CdFuturesConfigureEntity.java | 148 +++ .../java/com/msdw/tms/entity/CdPoint.java | 35 + .../com/msdw/tms/entity/CdPointChildren.java | 39 + .../CdTransferWarehouseRecordEntity.java | 52 + .../msdw/tms/entity/CdUserAssetsEntity.java | 50 + .../tms/entity/CdUserOptionAccountEntity.java | 56 + .../tms/entity/QhscGangjiaosuoEntity.java | 79 ++ .../entity/WhscRenminbipinzhongEntity.java | 190 ++++ .../request/ProjectRecordImportRequest.java | 41 + .../entity/request/QuestionsAddRequest.java | 67 ++ .../request/QuestionsImportRequest.java | 81 ++ .../entity/request/QuestionsQueryRequest.java | 39 + .../request/QuestionsUpdateRequest.java | 70 ++ .../msdw/tms/entity/response/CommonCode.java | 51 + .../msdw/tms/entity/response/Response.java | 9 + .../tms/entity/response/ResponseResult.java | 35 + .../msdw/tms/entity/response/ResultCode.java | 21 + .../msdw/tms/entity/vo/CdCreatePhotoVo.java | 19 + .../tms/entity/vo/CdExperimentalEntityVo.java | 59 ++ .../entity/vo/CdFuturesConfigureEntityVo.java | 73 ++ .../tms/feign/UserEntityFeignService.java | 13 + .../tms/interceptor/AuthorizedAspect.java | 92 ++ .../com/msdw/tms/model/req/CdBuyInReq.java | 21 + .../msdw/tms/model/req/CdExperimentReq.java | 62 ++ .../com/msdw/tms/model/req/CdExportReq.java | 31 + .../tms/model/req/CdGetAccountInfoReq.java | 40 + .../msdw/tms/model/req/CdGetCapitalReq.java | 11 + .../model/req/CdGetWeekAndMonthInfoReq.java | 23 + .../msdw/tms/model/req/CdOpenPositionReq.java | 41 + .../tms/model/req/CdPageConditionReq.java | 38 + .../msdw/tms/model/req/CdSaveCacheReq.java | 25 + .../msdw/tms/model/req/CdSimulationReq.java | 35 + .../tms/model/req/CdTransferRecordsReq.java | 39 + .../com/msdw/tms/model/req/CdTransferReq.java | 22 + .../tms/model/req/CdTransferWarehouseReq.java | 37 + .../tms/model/req/QhscGangjiaosuoReq.java | 38 + .../java/com/msdw/tms/model/req/Testreq.java | 14 + .../tms/model/resp/CdAccountInfoResp.java | 67 ++ .../tms/model/resp/CdAfterHedgingResp.java | 45 + .../tms/model/resp/CdBeforeHedgingResp.java | 73 ++ .../com/msdw/tms/model/resp/CdFxSpotResp.java | 83 ++ .../tms/model/resp/CdGetDetailByIdResp.java | 54 + .../tms/model/resp/CdGetExchangeRateResp.java | 21 + .../model/resp/CdGetWeekAndMonthInfoResp.java | 29 + .../model/resp/CdHedgingEfficiencyResp.java | 53 + .../msdw/tms/model/resp/CdHedgingResp.java | 33 + .../model/resp/CdHoldPositionInfoResp.java | 67 ++ .../tms/model/resp/CdTransferOutResp.java | 36 + .../tms/model/resp/CdTransferRecordsResp.java | 39 + .../resp/CdTransferWarehouseDetailResp.java | 86 ++ .../com/msdw/tms/model/resp/TestResq.java | 14 + .../msdw/tms/service/AliyunOssService.java | 29 + .../tms/service/CdBankStatementsService.java | 29 + .../com/msdw/tms/service/CdCacheService.java | 16 + .../tms/service/CdExperimentalService.java | 28 + .../service/CdFuturesConfigureService.java | 47 + .../tms/service/CdPointChildrenService.java | 14 + .../com/msdw/tms/service/CdPointService.java | 14 + .../CdTransferWarehouseRecordService.java | 17 + .../msdw/tms/service/CdUserAssetsService.java | 21 + .../service/CdUserOptionAccountService.java | 21 + .../tms/service/QhscGangjiaosuoService.java | 23 + .../com/msdw/tms/service/RedisService.java | 22 + .../service/WhscRenminbipinzhongService.java | 34 + .../service/impl/AliyunOssServiceImpl.java | 56 + .../impl/CdBankStatementsServiceImpl.java | 358 +++++++ .../tms/service/impl/CdCacheServiceImpl.java | 70 ++ .../impl/CdExperimentalServiceImpl.java | 259 +++++ .../impl/CdFuturesConfigureServiceImpl.java | 964 ++++++++++++++++++ .../impl/CdPointChildrenServiceImpl.java | 24 + .../tms/service/impl/CdPointServiceImpl.java | 24 + .../CdTransferWarehouseRecordServiceImpl.java | 366 +++++++ .../service/impl/CdUserAssetsServiceImpl.java | 66 ++ .../impl/CdUserOptionAccountServiceImpl.java | 73 ++ .../impl/QhscGangjiaosuoServiceImpl.java | 43 + .../impl/WhscRenminbipinzhongServiceImpl.java | 348 +++++++ src/main/resources/aliyun.properties | 6 + src/main/resources/application.yml | 61 ++ .../excel-template/实验报告导出.xlsx | Bin 0 -> 10295 bytes .../实验报告导出模板.xlsx | Bin 0 -> 10365 bytes .../试题导入失败数据导出模板.xlsx | Bin 0 -> 10358 bytes .../excel-template/试题导入模板.xlsx | Bin 0 -> 11147 bytes .../excel-template/试题导出模板.xlsx | Bin 0 -> 10311 bytes src/main/resources/logback-spring.xml | 47 + .../mapper/tms/CdBankStatementsMapper.xml | 23 + .../resources/mapper/tms/CdCacheMapper.xml | 22 + .../mapper/tms/CdExperimentalMapper.xml | 22 + .../mapper/tms/CdFuturesConfigureMapper.xml | 28 + .../mapper/tms/CdPointChildrenMapper.xml | 13 + .../resources/mapper/tms/CdPointMapper.xml | 12 + .../tms/CdTransferWarehouseRecordMapper.xml | 18 + .../mapper/tms/CdUserAssetsMapper.xml | 15 + .../mapper/tms/CdUserOptionAccountMapper.xml | 16 + .../mapper/tms/QhscGangjiaosuoMapper.xml | 49 + .../mapper/tms/WhscRenminbipinzhongMapper.xml | 24 + src/main/resources/sql/data.sql | 115 +++ src/test/java/com/msdw/tms/IdWorker.java | 146 +++ .../java/com/msdw/tms/SnowflakeIdUtils.java | 174 ++++ .../com/msdw/tms/TmsApplicationTests.java | 518 ++++++++++ src/test/java/com/msdw/tms/UserTest.java | 26 + .../tms/service/AliyunOssServiceTest.java | 62 ++ 172 files changed, 13452 insertions(+) create mode 100644 .gitignore create mode 100644 .mvn/wrapper/MavenWrapperDownloader.java create mode 100644 .mvn/wrapper/maven-wrapper.jar create mode 100644 .mvn/wrapper/maven-wrapper.properties create mode 100644 README.md create mode 100644 mvnw create mode 100644 mvnw.cmd create mode 100644 pom.xml create mode 100644 src/main/java/com/msdw/tms/TmsApplication.java create mode 100644 src/main/java/com/msdw/tms/aop/AuthCustomize.java create mode 100644 src/main/java/com/msdw/tms/common/Test.java create mode 100644 src/main/java/com/msdw/tms/common/exception/CustomException.java create mode 100644 src/main/java/com/msdw/tms/common/exception/ExceptionCast.java create mode 100644 src/main/java/com/msdw/tms/common/exception/ExceptionCatch.java create mode 100644 src/main/java/com/msdw/tms/common/exception/RRException.java create mode 100644 src/main/java/com/msdw/tms/common/utils/AliyunOssUtil.java create mode 100644 src/main/java/com/msdw/tms/common/utils/BasePage.java create mode 100644 src/main/java/com/msdw/tms/common/utils/Constant.java create mode 100644 src/main/java/com/msdw/tms/common/utils/DownloadUtils.java create mode 100644 src/main/java/com/msdw/tms/common/utils/FilesResult.java create mode 100644 src/main/java/com/msdw/tms/common/utils/MoneyConversionUtil.java create mode 100644 src/main/java/com/msdw/tms/common/utils/PageUtil.java create mode 100644 src/main/java/com/msdw/tms/common/utils/PageUtils.java create mode 100644 src/main/java/com/msdw/tms/common/utils/Query.java create mode 100644 src/main/java/com/msdw/tms/common/utils/R.java create mode 100644 src/main/java/com/msdw/tms/common/utils/RedisConfig.java create mode 100644 src/main/java/com/msdw/tms/common/utils/RestUtil.java create mode 100644 src/main/java/com/msdw/tms/common/utils/TokenUtil.java create mode 100644 src/main/java/com/msdw/tms/common/utils/date/DateUtils.java create mode 100644 src/main/java/com/msdw/tms/common/utils/photo/photoUrl.java create mode 100644 src/main/java/com/msdw/tms/common/utils/poi/EasyExcelUtil.java create mode 100644 src/main/java/com/msdw/tms/common/utils/poi/ExcelAttribute.java create mode 100644 src/main/java/com/msdw/tms/common/utils/poi/ExcelExportUtil.java create mode 100644 src/main/java/com/msdw/tms/common/utils/poi/ExcelImportUtil.java create mode 100644 src/main/java/com/msdw/tms/common/utils/py/PyUtil.java create mode 100644 src/main/java/com/msdw/tms/common/utils/trading_rules/TradingDay.java create mode 100644 src/main/java/com/msdw/tms/common/utils/trading_rules/algorithm.java create mode 100644 src/main/java/com/msdw/tms/common/xss/HTMLFilter.java create mode 100644 src/main/java/com/msdw/tms/common/xss/SQLFilter.java create mode 100644 src/main/java/com/msdw/tms/config/AliyunOssConfig.java create mode 100644 src/main/java/com/msdw/tms/config/CorsConfig.java create mode 100644 src/main/java/com/msdw/tms/config/MyThreadConfig.java create mode 100644 src/main/java/com/msdw/tms/config/MybatisPlusConfig.java create mode 100644 src/main/java/com/msdw/tms/config/RedisConfig.java create mode 100644 src/main/java/com/msdw/tms/config/SwaggerConfig.java create mode 100644 src/main/java/com/msdw/tms/config/ThreadPoolConfigProperties.java create mode 100644 src/main/java/com/msdw/tms/config/WebConfigurer.java create mode 100644 src/main/java/com/msdw/tms/controller/CdBankStatementsController.java create mode 100644 src/main/java/com/msdw/tms/controller/CdCacheController.java create mode 100644 src/main/java/com/msdw/tms/controller/CdExperimentalController.java create mode 100644 src/main/java/com/msdw/tms/controller/CdFuturesConfigureController.java create mode 100644 src/main/java/com/msdw/tms/controller/CdHedgingController.java create mode 100644 src/main/java/com/msdw/tms/controller/CdPointController.java create mode 100644 src/main/java/com/msdw/tms/controller/CdTransferWarehouseRecordController.java create mode 100644 src/main/java/com/msdw/tms/controller/CdUserAssetsController.java create mode 100644 src/main/java/com/msdw/tms/controller/QhscGangjiaosuoController.java create mode 100644 src/main/java/com/msdw/tms/controller/WhscRenminbipinzhongController.java create mode 100644 src/main/java/com/msdw/tms/dao/CdBankStatementsDao.java create mode 100644 src/main/java/com/msdw/tms/dao/CdCacheDao.java create mode 100644 src/main/java/com/msdw/tms/dao/CdExperimentalDao.java create mode 100644 src/main/java/com/msdw/tms/dao/CdFuturesConfigureDao.java create mode 100644 src/main/java/com/msdw/tms/dao/CdPointChildrenDao.java create mode 100644 src/main/java/com/msdw/tms/dao/CdPointDao.java create mode 100644 src/main/java/com/msdw/tms/dao/CdTransferWarehouseRecordDao.java create mode 100644 src/main/java/com/msdw/tms/dao/CdUserAssetsDao.java create mode 100644 src/main/java/com/msdw/tms/dao/CdUserOptionAccountDao.java create mode 100644 src/main/java/com/msdw/tms/dao/QhscGangjiaosuoDao.java create mode 100644 src/main/java/com/msdw/tms/dao/WhscRenminbipinzhongDao.java create mode 100644 src/main/java/com/msdw/tms/entity/CdBankStatementsEntity.java create mode 100644 src/main/java/com/msdw/tms/entity/CdCacheEntity.java create mode 100644 src/main/java/com/msdw/tms/entity/CdExperimentalEntity.java create mode 100644 src/main/java/com/msdw/tms/entity/CdFuturesConfigureEntity.java create mode 100644 src/main/java/com/msdw/tms/entity/CdPoint.java create mode 100644 src/main/java/com/msdw/tms/entity/CdPointChildren.java create mode 100644 src/main/java/com/msdw/tms/entity/CdTransferWarehouseRecordEntity.java create mode 100644 src/main/java/com/msdw/tms/entity/CdUserAssetsEntity.java create mode 100644 src/main/java/com/msdw/tms/entity/CdUserOptionAccountEntity.java create mode 100644 src/main/java/com/msdw/tms/entity/QhscGangjiaosuoEntity.java create mode 100644 src/main/java/com/msdw/tms/entity/WhscRenminbipinzhongEntity.java create mode 100644 src/main/java/com/msdw/tms/entity/request/ProjectRecordImportRequest.java create mode 100644 src/main/java/com/msdw/tms/entity/request/QuestionsAddRequest.java create mode 100644 src/main/java/com/msdw/tms/entity/request/QuestionsImportRequest.java create mode 100644 src/main/java/com/msdw/tms/entity/request/QuestionsQueryRequest.java create mode 100644 src/main/java/com/msdw/tms/entity/request/QuestionsUpdateRequest.java create mode 100644 src/main/java/com/msdw/tms/entity/response/CommonCode.java create mode 100644 src/main/java/com/msdw/tms/entity/response/Response.java create mode 100644 src/main/java/com/msdw/tms/entity/response/ResponseResult.java create mode 100644 src/main/java/com/msdw/tms/entity/response/ResultCode.java create mode 100644 src/main/java/com/msdw/tms/entity/vo/CdCreatePhotoVo.java create mode 100644 src/main/java/com/msdw/tms/entity/vo/CdExperimentalEntityVo.java create mode 100644 src/main/java/com/msdw/tms/entity/vo/CdFuturesConfigureEntityVo.java create mode 100644 src/main/java/com/msdw/tms/feign/UserEntityFeignService.java create mode 100644 src/main/java/com/msdw/tms/interceptor/AuthorizedAspect.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdBuyInReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdExperimentReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdExportReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdGetAccountInfoReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdGetCapitalReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdGetWeekAndMonthInfoReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdOpenPositionReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdPageConditionReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdSaveCacheReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdSimulationReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdTransferRecordsReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdTransferReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/CdTransferWarehouseReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/QhscGangjiaosuoReq.java create mode 100644 src/main/java/com/msdw/tms/model/req/Testreq.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdAccountInfoResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdAfterHedgingResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdBeforeHedgingResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdFxSpotResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdGetDetailByIdResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdGetExchangeRateResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdGetWeekAndMonthInfoResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdHedgingEfficiencyResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdHedgingResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdHoldPositionInfoResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdTransferOutResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdTransferRecordsResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/CdTransferWarehouseDetailResp.java create mode 100644 src/main/java/com/msdw/tms/model/resp/TestResq.java create mode 100644 src/main/java/com/msdw/tms/service/AliyunOssService.java create mode 100644 src/main/java/com/msdw/tms/service/CdBankStatementsService.java create mode 100644 src/main/java/com/msdw/tms/service/CdCacheService.java create mode 100644 src/main/java/com/msdw/tms/service/CdExperimentalService.java create mode 100644 src/main/java/com/msdw/tms/service/CdFuturesConfigureService.java create mode 100644 src/main/java/com/msdw/tms/service/CdPointChildrenService.java create mode 100644 src/main/java/com/msdw/tms/service/CdPointService.java create mode 100644 src/main/java/com/msdw/tms/service/CdTransferWarehouseRecordService.java create mode 100644 src/main/java/com/msdw/tms/service/CdUserAssetsService.java create mode 100644 src/main/java/com/msdw/tms/service/CdUserOptionAccountService.java create mode 100644 src/main/java/com/msdw/tms/service/QhscGangjiaosuoService.java create mode 100644 src/main/java/com/msdw/tms/service/RedisService.java create mode 100644 src/main/java/com/msdw/tms/service/WhscRenminbipinzhongService.java create mode 100644 src/main/java/com/msdw/tms/service/impl/AliyunOssServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/CdBankStatementsServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/CdCacheServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/CdExperimentalServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/CdFuturesConfigureServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/CdPointChildrenServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/CdPointServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/CdTransferWarehouseRecordServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/CdUserAssetsServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/CdUserOptionAccountServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/QhscGangjiaosuoServiceImpl.java create mode 100644 src/main/java/com/msdw/tms/service/impl/WhscRenminbipinzhongServiceImpl.java create mode 100644 src/main/resources/aliyun.properties create mode 100644 src/main/resources/application.yml create mode 100644 src/main/resources/excel-template/实验报告导出.xlsx create mode 100644 src/main/resources/excel-template/实验报告导出模板.xlsx create mode 100644 src/main/resources/excel-template/试题导入失败数据导出模板.xlsx create mode 100644 src/main/resources/excel-template/试题导入模板.xlsx create mode 100644 src/main/resources/excel-template/试题导出模板.xlsx create mode 100644 src/main/resources/logback-spring.xml create mode 100644 src/main/resources/mapper/tms/CdBankStatementsMapper.xml create mode 100644 src/main/resources/mapper/tms/CdCacheMapper.xml create mode 100644 src/main/resources/mapper/tms/CdExperimentalMapper.xml create mode 100644 src/main/resources/mapper/tms/CdFuturesConfigureMapper.xml create mode 100644 src/main/resources/mapper/tms/CdPointChildrenMapper.xml create mode 100644 src/main/resources/mapper/tms/CdPointMapper.xml create mode 100644 src/main/resources/mapper/tms/CdTransferWarehouseRecordMapper.xml create mode 100644 src/main/resources/mapper/tms/CdUserAssetsMapper.xml create mode 100644 src/main/resources/mapper/tms/CdUserOptionAccountMapper.xml create mode 100644 src/main/resources/mapper/tms/QhscGangjiaosuoMapper.xml create mode 100644 src/main/resources/mapper/tms/WhscRenminbipinzhongMapper.xml create mode 100644 src/main/resources/sql/data.sql create mode 100644 src/test/java/com/msdw/tms/IdWorker.java create mode 100644 src/test/java/com/msdw/tms/SnowflakeIdUtils.java create mode 100644 src/test/java/com/msdw/tms/TmsApplicationTests.java create mode 100644 src/test/java/com/msdw/tms/UserTest.java create mode 100644 src/test/java/com/msdw/tms/service/AliyunOssServiceTest.java diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..549e00a --- /dev/null +++ b/.gitignore @@ -0,0 +1,33 @@ +HELP.md +target/ +!.mvn/wrapper/maven-wrapper.jar +!**/src/main/**/target/ +!**/src/test/**/target/ + +### STS ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache + +### IntelliJ IDEA ### +.idea +*.iws +*.iml +*.ipr + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ +build/ +!**/src/main/**/build/ +!**/src/test/**/build/ + +### VS Code ### +.vscode/ diff --git a/.mvn/wrapper/MavenWrapperDownloader.java b/.mvn/wrapper/MavenWrapperDownloader.java new file mode 100644 index 0000000..a45eb6b --- /dev/null +++ b/.mvn/wrapper/MavenWrapperDownloader.java @@ -0,0 +1,118 @@ +/* + * Copyright 2007-present the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import java.net.*; +import java.io.*; +import java.nio.channels.*; +import java.util.Properties; + +public class MavenWrapperDownloader { + + private static final String WRAPPER_VERSION = "0.5.6"; + /** + * Default URL to download the maven-wrapper.jar from, if no 'downloadUrl' is provided. + */ + private static final String DEFAULT_DOWNLOAD_URL = "https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/" + + WRAPPER_VERSION + "/maven-wrapper-" + WRAPPER_VERSION + ".jar"; + + /** + * Path to the maven-wrapper.properties file, which might contain a downloadUrl property to + * use instead of the default one. + */ + private static final String MAVEN_WRAPPER_PROPERTIES_PATH = + ".mvn/wrapper/maven-wrapper.properties"; + + /** + * Path where the maven-wrapper.jar will be saved to. + */ + private static final String MAVEN_WRAPPER_JAR_PATH = + ".mvn/wrapper/maven-wrapper.jar"; + + /** + * Name of the property which should be used to override the default download url for the wrapper. + */ + private static final String PROPERTY_NAME_WRAPPER_URL = "wrapperUrl"; + + public static void main(String args[]) { + System.out.println("- Downloader started"); + File baseDirectory = new File(args[0]); + System.out.println("- Using base directory: " + baseDirectory.getAbsolutePath()); + + // If the maven-wrapper.properties exists, read it and check if it contains a custom + // wrapperUrl parameter. + File mavenWrapperPropertyFile = new File(baseDirectory, MAVEN_WRAPPER_PROPERTIES_PATH); + String url = DEFAULT_DOWNLOAD_URL; + if (mavenWrapperPropertyFile.exists()) { + FileInputStream mavenWrapperPropertyFileInputStream = null; + try { + mavenWrapperPropertyFileInputStream = new FileInputStream(mavenWrapperPropertyFile); + Properties mavenWrapperProperties = new Properties(); + mavenWrapperProperties.load(mavenWrapperPropertyFileInputStream); + url = mavenWrapperProperties.getProperty(PROPERTY_NAME_WRAPPER_URL, url); + } catch (IOException e) { + System.out.println("- ERROR loading '" + MAVEN_WRAPPER_PROPERTIES_PATH + "'"); + } finally { + try { + if (mavenWrapperPropertyFileInputStream != null) { + mavenWrapperPropertyFileInputStream.close(); + } + } catch (IOException e) { + // Ignore ... + } + } + } + System.out.println("- Downloading from: " + url); + + File outputFile = new File(baseDirectory.getAbsolutePath(), MAVEN_WRAPPER_JAR_PATH); + if (!outputFile.getParentFile().exists()) { + if (!outputFile.getParentFile().mkdirs()) { + System.out.println( + "- ERROR creating output directory '" + outputFile.getParentFile().getAbsolutePath() + "'"); + } + } + System.out.println("- Downloading to: " + outputFile.getAbsolutePath()); + try { + downloadFileFromURL(url, outputFile); + System.out.println("Done"); + System.exit(0); + } catch (Throwable e) { + System.out.println("- Error downloading"); + e.printStackTrace(); + System.exit(1); + } + } + + private static void downloadFileFromURL(String urlString, File destination) throws Exception { + if (System.getenv("MVNW_USERNAME") != null && System.getenv("MVNW_PASSWORD") != null) { + String username = System.getenv("MVNW_USERNAME"); + char[] password = System.getenv("MVNW_PASSWORD").toCharArray(); + Authenticator.setDefault(new Authenticator() { + @Override + protected PasswordAuthentication getPasswordAuthentication() { + return new PasswordAuthentication(username, password); + } + }); + } + URL website = new URL(urlString); + ReadableByteChannel rbc; + rbc = Channels.newChannel(website.openStream()); + FileOutputStream fos = new FileOutputStream(destination); + fos.getChannel().transferFrom(rbc, 0, Long.MAX_VALUE); + fos.close(); + rbc.close(); + } + +} diff --git a/.mvn/wrapper/maven-wrapper.jar b/.mvn/wrapper/maven-wrapper.jar new file mode 100644 index 0000000000000000000000000000000000000000..2cc7d4a55c0cd0092912bf49ae38b3a9e3fd0054 GIT binary patch literal 50710 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 EfAhh=DF6Tf literal 0 HcmV?d00001 diff --git a/.mvn/wrapper/maven-wrapper.properties b/.mvn/wrapper/maven-wrapper.properties new file mode 100644 index 0000000..642d572 --- /dev/null +++ b/.mvn/wrapper/maven-wrapper.properties @@ -0,0 +1,2 @@ +distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.6.3/apache-maven-3.6.3-bin.zip +wrapperUrl=https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar diff --git a/README.md b/README.md new file mode 100644 index 0000000..107f8bf --- /dev/null +++ b/README.md @@ -0,0 +1,3 @@ +# msdw_tms + +测评管理系统 \ No newline at end of file diff --git a/mvnw b/mvnw new file mode 100644 index 0000000..a16b543 --- /dev/null +++ b/mvnw @@ -0,0 +1,310 @@ +#!/bin/sh +# ---------------------------------------------------------------------------- +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# ---------------------------------------------------------------------------- + +# ---------------------------------------------------------------------------- +# Maven Start Up Batch script +# +# Required ENV vars: +# ------------------ +# JAVA_HOME - location of a JDK home dir +# +# Optional ENV vars +# ----------------- +# M2_HOME - location of maven2's installed home dir +# MAVEN_OPTS - parameters passed to the Java VM when running Maven +# e.g. to debug Maven itself, use +# set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +# MAVEN_SKIP_RC - flag to disable loading of mavenrc files +# ---------------------------------------------------------------------------- + +if [ -z "$MAVEN_SKIP_RC" ] ; then + + if [ -f /etc/mavenrc ] ; then + . /etc/mavenrc + fi + + if [ -f "$HOME/.mavenrc" ] ; then + . "$HOME/.mavenrc" + fi + +fi + +# OS specific support. $var _must_ be set to either true or false. +cygwin=false; +darwin=false; +mingw=false +case "`uname`" in + CYGWIN*) cygwin=true ;; + MINGW*) mingw=true;; + Darwin*) darwin=true + # Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home + # See https://developer.apple.com/library/mac/qa/qa1170/_index.html + if [ -z "$JAVA_HOME" ]; then + if [ -x "/usr/libexec/java_home" ]; then + export JAVA_HOME="`/usr/libexec/java_home`" + else + export JAVA_HOME="/Library/Java/Home" + fi + fi + ;; +esac + +if [ -z "$JAVA_HOME" ] ; then + if [ -r /etc/gentoo-release ] ; then + JAVA_HOME=`java-config --jre-home` + fi +fi + +if [ -z "$M2_HOME" ] ; then + ## resolve links - $0 may be a link to maven's home + PRG="$0" + + # need this for relative symlinks + while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG="`dirname "$PRG"`/$link" + fi + done + + saveddir=`pwd` + + M2_HOME=`dirname "$PRG"`/.. + + # make it fully qualified + M2_HOME=`cd "$M2_HOME" && pwd` + + cd "$saveddir" + # echo Using m2 at $M2_HOME +fi + +# For Cygwin, ensure paths are in UNIX format before anything is touched +if $cygwin ; then + [ -n "$M2_HOME" ] && + M2_HOME=`cygpath --unix "$M2_HOME"` + [ -n "$JAVA_HOME" ] && + JAVA_HOME=`cygpath --unix "$JAVA_HOME"` + [ -n "$CLASSPATH" ] && + CLASSPATH=`cygpath --path --unix "$CLASSPATH"` +fi + +# For Mingw, ensure paths are in UNIX format before anything is touched +if $mingw ; then + [ -n "$M2_HOME" ] && + M2_HOME="`(cd "$M2_HOME"; pwd)`" + [ -n "$JAVA_HOME" ] && + JAVA_HOME="`(cd "$JAVA_HOME"; pwd)`" +fi + +if [ -z "$JAVA_HOME" ]; then + javaExecutable="`which javac`" + if [ -n "$javaExecutable" ] && ! [ "`expr \"$javaExecutable\" : '\([^ ]*\)'`" = "no" ]; then + # readlink(1) is not available as standard on Solaris 10. + readLink=`which readlink` + if [ ! `expr "$readLink" : '\([^ ]*\)'` = "no" ]; then + if $darwin ; then + javaHome="`dirname \"$javaExecutable\"`" + javaExecutable="`cd \"$javaHome\" && pwd -P`/javac" + else + javaExecutable="`readlink -f \"$javaExecutable\"`" + fi + javaHome="`dirname \"$javaExecutable\"`" + javaHome=`expr "$javaHome" : '\(.*\)/bin'` + JAVA_HOME="$javaHome" + export JAVA_HOME + fi + fi +fi + +if [ -z "$JAVACMD" ] ; then + if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + else + JAVACMD="`which java`" + fi +fi + +if [ ! -x "$JAVACMD" ] ; then + echo "Error: JAVA_HOME is not defined correctly." >&2 + echo " We cannot execute $JAVACMD" >&2 + exit 1 +fi + +if [ -z "$JAVA_HOME" ] ; then + echo "Warning: JAVA_HOME environment variable is not set." +fi + +CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher + +# traverses directory structure from process work directory to filesystem root +# first directory with .mvn subdirectory is considered project base directory +find_maven_basedir() { + + if [ -z "$1" ] + then + echo "Path not specified to find_maven_basedir" + return 1 + fi + + basedir="$1" + wdir="$1" + while [ "$wdir" != '/' ] ; do + if [ -d "$wdir"/.mvn ] ; then + basedir=$wdir + break + fi + # workaround for JBEAP-8937 (on Solaris 10/Sparc) + if [ -d "${wdir}" ]; then + wdir=`cd "$wdir/.."; pwd` + fi + # end of workaround + done + echo "${basedir}" +} + +# concatenates all lines of a file +concat_lines() { + if [ -f "$1" ]; then + echo "$(tr -s '\n' ' ' < "$1")" + fi +} + +BASE_DIR=`find_maven_basedir "$(pwd)"` +if [ -z "$BASE_DIR" ]; then + exit 1; +fi + +########################################################################################## +# Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +# This allows using the maven wrapper in projects that prohibit checking in binary data. +########################################################################################## +if [ -r "$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found .mvn/wrapper/maven-wrapper.jar" + fi +else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..." + fi + if [ -n "$MVNW_REPOURL" ]; then + jarUrl="$MVNW_REPOURL/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + else + jarUrl="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + fi + while IFS="=" read key value; do + case "$key" in (wrapperUrl) jarUrl="$value"; break ;; + esac + done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties" + if [ "$MVNW_VERBOSE" = true ]; then + echo "Downloading from: $jarUrl" + fi + wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" + if $cygwin; then + wrapperJarPath=`cygpath --path --windows "$wrapperJarPath"` + fi + + if command -v wget > /dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found wget ... using wget" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + wget "$jarUrl" -O "$wrapperJarPath" + else + wget --http-user=$MVNW_USERNAME --http-password=$MVNW_PASSWORD "$jarUrl" -O "$wrapperJarPath" + fi + elif command -v curl > /dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found curl ... using curl" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + curl -o "$wrapperJarPath" "$jarUrl" -f + else + curl --user $MVNW_USERNAME:$MVNW_PASSWORD -o "$wrapperJarPath" "$jarUrl" -f + fi + + else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Falling back to using Java to download" + fi + javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java" + # For Cygwin, switch paths to Windows format before running javac + if $cygwin; then + javaClass=`cygpath --path --windows "$javaClass"` + fi + if [ -e "$javaClass" ]; then + if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Compiling MavenWrapperDownloader.java ..." + fi + # Compiling the Java class + ("$JAVA_HOME/bin/javac" "$javaClass") + fi + if [ -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + # Running the downloader + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Running MavenWrapperDownloader.java ..." + fi + ("$JAVA_HOME/bin/java" -cp .mvn/wrapper MavenWrapperDownloader "$MAVEN_PROJECTBASEDIR") + fi + fi + fi +fi +########################################################################################## +# End of extension +########################################################################################## + +export MAVEN_PROJECTBASEDIR=${MAVEN_BASEDIR:-"$BASE_DIR"} +if [ "$MVNW_VERBOSE" = true ]; then + echo $MAVEN_PROJECTBASEDIR +fi +MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS" + +# For Cygwin, switch paths to Windows format before running java +if $cygwin; then + [ -n "$M2_HOME" ] && + M2_HOME=`cygpath --path --windows "$M2_HOME"` + [ -n "$JAVA_HOME" ] && + JAVA_HOME=`cygpath --path --windows "$JAVA_HOME"` + [ -n "$CLASSPATH" ] && + CLASSPATH=`cygpath --path --windows "$CLASSPATH"` + [ -n "$MAVEN_PROJECTBASEDIR" ] && + MAVEN_PROJECTBASEDIR=`cygpath --path --windows "$MAVEN_PROJECTBASEDIR"` +fi + +# Provide a "standardized" way to retrieve the CLI args that will +# work with both Windows and non-Windows executions. +MAVEN_CMD_LINE_ARGS="$MAVEN_CONFIG $@" +export MAVEN_CMD_LINE_ARGS + +WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +exec "$JAVACMD" \ + $MAVEN_OPTS \ + -classpath "$MAVEN_PROJECTBASEDIR/.mvn/wrapper/maven-wrapper.jar" \ + "-Dmaven.home=${M2_HOME}" "-Dmaven.multiModuleProjectDirectory=${MAVEN_PROJECTBASEDIR}" \ + ${WRAPPER_LAUNCHER} $MAVEN_CONFIG "$@" diff --git a/mvnw.cmd b/mvnw.cmd new file mode 100644 index 0000000..c8d4337 --- /dev/null +++ b/mvnw.cmd @@ -0,0 +1,182 @@ +@REM ---------------------------------------------------------------------------- +@REM Licensed to the Apache Software Foundation (ASF) under one +@REM or more contributor license agreements. See the NOTICE file +@REM distributed with this work for additional information +@REM regarding copyright ownership. The ASF licenses this file +@REM to you under the Apache License, Version 2.0 (the +@REM "License"); you may not use this file except in compliance +@REM with the License. You may obtain a copy of the License at +@REM +@REM https://www.apache.org/licenses/LICENSE-2.0 +@REM +@REM Unless required by applicable law or agreed to in writing, +@REM software distributed under the License is distributed on an +@REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +@REM KIND, either express or implied. See the License for the +@REM specific language governing permissions and limitations +@REM under the License. +@REM ---------------------------------------------------------------------------- + +@REM ---------------------------------------------------------------------------- +@REM Maven Start Up Batch script +@REM +@REM Required ENV vars: +@REM JAVA_HOME - location of a JDK home dir +@REM +@REM Optional ENV vars +@REM M2_HOME - location of maven2's installed home dir +@REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands +@REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a keystroke before ending +@REM MAVEN_OPTS - parameters passed to the Java VM when running Maven +@REM e.g. to debug Maven itself, use +@REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +@REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files +@REM ---------------------------------------------------------------------------- + +@REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on' +@echo off +@REM set title of command window +title %0 +@REM enable echoing by setting MAVEN_BATCH_ECHO to 'on' +@if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO% + +@REM set %HOME% to equivalent of $HOME +if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%") + +@REM Execute a user defined script before this one +if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre +@REM check for pre script, once with legacy .bat ending and once with .cmd ending +if exist "%HOME%\mavenrc_pre.bat" call "%HOME%\mavenrc_pre.bat" +if exist "%HOME%\mavenrc_pre.cmd" call "%HOME%\mavenrc_pre.cmd" +:skipRcPre + +@setlocal + +set ERROR_CODE=0 + +@REM To isolate internal variables from possible post scripts, we use another setlocal +@setlocal + +@REM ==== START VALIDATION ==== +if not "%JAVA_HOME%" == "" goto OkJHome + +echo. +echo Error: JAVA_HOME not found in your environment. >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +:OkJHome +if exist "%JAVA_HOME%\bin\java.exe" goto init + +echo. +echo Error: JAVA_HOME is set to an invalid directory. >&2 +echo JAVA_HOME = "%JAVA_HOME%" >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +@REM ==== END VALIDATION ==== + +:init + +@REM Find the project base dir, i.e. the directory that contains the folder ".mvn". +@REM Fallback to current working directory if not found. + +set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR% +IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir + +set EXEC_DIR=%CD% +set WDIR=%EXEC_DIR% +:findBaseDir +IF EXIST "%WDIR%"\.mvn goto baseDirFound +cd .. +IF "%WDIR%"=="%CD%" goto baseDirNotFound +set WDIR=%CD% +goto findBaseDir + +:baseDirFound +set MAVEN_PROJECTBASEDIR=%WDIR% +cd "%EXEC_DIR%" +goto endDetectBaseDir + +:baseDirNotFound +set MAVEN_PROJECTBASEDIR=%EXEC_DIR% +cd "%EXEC_DIR%" + +:endDetectBaseDir + +IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig + +@setlocal EnableExtensions EnableDelayedExpansion +for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a +@endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS% + +:endReadAdditionalConfig + +SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe" +set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar" +set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + +FOR /F "tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO ( + IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B +) + +@REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +@REM This allows using the maven wrapper in projects that prohibit checking in binary data. +if exist %WRAPPER_JAR% ( + if "%MVNW_VERBOSE%" == "true" ( + echo Found %WRAPPER_JAR% + ) +) else ( + if not "%MVNW_REPOURL%" == "" ( + SET DOWNLOAD_URL="%MVNW_REPOURL%/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + ) + if "%MVNW_VERBOSE%" == "true" ( + echo Couldn't find %WRAPPER_JAR%, downloading it ... + echo Downloading from: %DOWNLOAD_URL% + ) + + powershell -Command "&{"^ + "$webclient = new-object System.Net.WebClient;"^ + "if (-not ([string]::IsNullOrEmpty('%MVNW_USERNAME%') -and [string]::IsNullOrEmpty('%MVNW_PASSWORD%'))) {"^ + "$webclient.Credentials = new-object System.Net.NetworkCredential('%MVNW_USERNAME%', '%MVNW_PASSWORD%');"^ + "}"^ + "[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; $webclient.DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"^ + "}" + if "%MVNW_VERBOSE%" == "true" ( + echo Finished downloading %WRAPPER_JAR% + ) +) +@REM End of extension + +@REM Provide a "standardized" way to retrieve the CLI args that will +@REM work with both Windows and non-Windows executions. +set MAVEN_CMD_LINE_ARGS=%* + +%MAVEN_JAVA_EXE% %JVM_CONFIG_MAVEN_PROPS% %MAVEN_OPTS% %MAVEN_DEBUG_OPTS% -classpath %WRAPPER_JAR% "-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" %WRAPPER_LAUNCHER% %MAVEN_CONFIG% %* +if ERRORLEVEL 1 goto error +goto end + +:error +set ERROR_CODE=1 + +:end +@endlocal & set ERROR_CODE=%ERROR_CODE% + +if not "%MAVEN_SKIP_RC%" == "" goto skipRcPost +@REM check for post script, once with legacy .bat ending and once with .cmd ending +if exist "%HOME%\mavenrc_post.bat" call "%HOME%\mavenrc_post.bat" +if exist "%HOME%\mavenrc_post.cmd" call "%HOME%\mavenrc_post.cmd" +:skipRcPost + +@REM pause the script if MAVEN_BATCH_PAUSE is set to 'on' +if "%MAVEN_BATCH_PAUSE%" == "on" pause + +if "%MAVEN_TERMINATE_CMD%" == "on" exit %ERROR_CODE% + +exit /B %ERROR_CODE% diff --git a/pom.xml b/pom.xml new file mode 100644 index 0000000..381f2dc --- /dev/null +++ b/pom.xml @@ -0,0 +1,236 @@ + + + 4.0.0 + + org.springframework.boot + spring-boot-starter-parent + 2.3.4.RELEASE + + + com.msdw + tms + 0.0.1-SNAPSHOT + tms + 测评管理系统 + + jar + + 1.8 + + + + + org.apache.poi + poi + 4.0.1 + + + org.apache.poi + poi-ooxml + 4.0.1 + + + org.apache.poi + poi-ooxml-schemas + 4.0.1 + + + + org.springframework.boot + spring-boot-starter-web + + + org.springframework.boot + spring-boot-starter-tomcat + + compile + + + + com.baomidou + mybatis-plus-boot-starter + 3.3.1 + + + org.projectlombok + lombok + 1.18.12 + + + + org.apache.httpcomponents + httpcore + 4.4.13 + + + + commons-lang + commons-lang + 2.6 + + + + + mysql + mysql-connector-java + 8.0.20 + + + + + io.springfox + springfox-swagger-ui + 2.7.0 + + + io.springfox + springfox-swagger2 + 2.7.0 + + + + com.aliyun.oss + aliyun-sdk-oss + 2.8.2 + + + org.springframework.cloud + spring-cloud-starter-openfeign + 2.2.3.RELEASE + + + org.springframework.boot + spring-boot-starter-test + test + + + org.junit.vintage + junit-vintage-engine + + + + + commons-io + commons-io + 2.6 + test + + + commons-fileupload + commons-fileupload + 1.4 + test + + + + org.springframework.boot + spring-boot-starter-validation + + + + org.springframework.boot + spring-boot-starter-data-redis + + + + cn.hutool + hutool-all + 5.3.8 + + + + com.alibaba + fastjson + 1.2.72 + + + org.springframework.boot + spring-boot-configuration-processor + true + + + + + org.python + jython-standalone + 2.7.0 + + + + + + + + cn.afterturn + easypoi-base + 4.1.3 + + + cn.afterturn + easypoi-web + 4.1.3 + + + cn.afterturn + easypoi-annotation + 4.1.3 + + + + com.alibaba + easyexcel + 2.1.1 + + + + + + + com.baomidou + dynamic-datasource-spring-boot-starter + 3.1.0 + + + + + org.springframework.boot + spring-boot-starter-aop + + + + + redis.clients + jedis + 2.9.0 + + + + org.springframework.boot + spring-boot-starter-data-redis + + + + + + + + evaluation + + + org.springframework.boot + spring-boot-maven-plugin + + + org.apache.maven.plugins + maven-surefire-plugin + 2.20.1 + + true + + + + + + diff --git a/src/main/java/com/msdw/tms/TmsApplication.java b/src/main/java/com/msdw/tms/TmsApplication.java new file mode 100644 index 0000000..e121ed0 --- /dev/null +++ b/src/main/java/com/msdw/tms/TmsApplication.java @@ -0,0 +1,20 @@ +package com.msdw.tms; + +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.SpringBootApplication; +import org.springframework.boot.builder.SpringApplicationBuilder; +import org.springframework.boot.web.servlet.support.SpringBootServletInitializer; + + +@SpringBootApplication +public class TmsApplication extends SpringBootServletInitializer { + + public static void main(String[] args) { + SpringApplication.run(TmsApplication.class, args); + } + + @Override + protected SpringApplicationBuilder configure(SpringApplicationBuilder builder) { + return builder.sources(TmsApplication.class); + } +} diff --git a/src/main/java/com/msdw/tms/aop/AuthCustomize.java b/src/main/java/com/msdw/tms/aop/AuthCustomize.java new file mode 100644 index 0000000..9d615db --- /dev/null +++ b/src/main/java/com/msdw/tms/aop/AuthCustomize.java @@ -0,0 +1,28 @@ +package com.msdw.tms.aop; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.aop + * @ClassName: Authorization + * @Description: 安全认证 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/22 9:46 + * @UpdateDate: 2020/10/22 9:46 + * @Version: 1.0 + */ + +import org.springframework.stereotype.Component; + +import java.lang.annotation.*; + +/** + * 在Controller的方法上使用此注解,该方法在映射时会检查用户鉴权 + */ +@Target(ElementType.METHOD) +@Retention(RetentionPolicy.RUNTIME) +@Documented +@Component +public @interface AuthCustomize { + //用于自定义描述等 + String value() default ""; +} diff --git a/src/main/java/com/msdw/tms/common/Test.java b/src/main/java/com/msdw/tms/common/Test.java new file mode 100644 index 0000000..558bd95 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/Test.java @@ -0,0 +1,21 @@ +package com.msdw.tms.common; + +import com.msdw.tms.common.utils.trading_rules.algorithm; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common + * @ClassName: Test + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/4 19:01 + * @UpdateDate: 2020/11/4 19:01 + * @Version: 1.0 + */ +public class Test { + + public static void main(String[] args) { + + System.out.println(algorithm.getLastTradingDay("美元兑人民币期货2010")); + } +} diff --git a/src/main/java/com/msdw/tms/common/exception/CustomException.java b/src/main/java/com/msdw/tms/common/exception/CustomException.java new file mode 100644 index 0000000..515fd6e --- /dev/null +++ b/src/main/java/com/msdw/tms/common/exception/CustomException.java @@ -0,0 +1,21 @@ +package com.msdw.tms.common.exception; + +import com.msdw.tms.entity.response.ResultCode; + +/** + * @author 世杰 + * @date 2020/3/24 22:47 + */ +public class CustomException extends RuntimeException { + private ResultCode resultCode; + + /*public CustomException(ResultCode resultCode) { + //异常信息为错误代码+异常信息 + super("错误代码:" + resultCode.code() + "错误信息:" + resultCode.message()); + this.resultCode = resultCode; + } + + public ResultCode getResultCode() { + return this.resultCode; + }*/ +} diff --git a/src/main/java/com/msdw/tms/common/exception/ExceptionCast.java b/src/main/java/com/msdw/tms/common/exception/ExceptionCast.java new file mode 100644 index 0000000..561bb1a --- /dev/null +++ b/src/main/java/com/msdw/tms/common/exception/ExceptionCast.java @@ -0,0 +1,12 @@ +package com.msdw.tms.common.exception; + +/** + * @author 世杰 + * @date 2020/3/24 23:04 + */ +public class ExceptionCast { + //使用此静态方法抛出自定义异常 + /*public static void cast(ResultCode resultCode) { + throw new CustomException(resultCode); + }*/ +} diff --git a/src/main/java/com/msdw/tms/common/exception/ExceptionCatch.java b/src/main/java/com/msdw/tms/common/exception/ExceptionCatch.java new file mode 100644 index 0000000..ecf7c18 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/exception/ExceptionCatch.java @@ -0,0 +1,53 @@ +package com.msdw.tms.common.exception; + +import org.springframework.web.bind.annotation.ControllerAdvice; + + +/** + * @author 世杰 + * @date 2020/3/24 23:06 + */ +//使用 @ControllerAdvice和@ExceptionHandler注解来捕获指定类型的异常 +@ControllerAdvice//控制器增强 +public class ExceptionCatch { + /* private static final Logger LOGGER = LoggerFactory.getLogger(ExceptionCatch.class); + + //定义Map,配置异常类型所对应的错误代码,使用 ImmutableMap 一旦写入数据,就无法更改,只读并且线程安全 + private static ImmutableMap, ResultCode> EXCEPTIONS; + //定义map的builder对象,去构建ImmutableMap + protected static ImmutableMap.Builder, ResultCode> builder = ImmutableMap.builder(); + + @ExceptionHandler(Exception.class) + @ResponseBody + public ResponseResult exception(Exception exception) { + exception.printStackTrace(); + //记录日志 + LOGGER.error("catch exception:{}", exception.getMessage()); + if (EXCEPTIONS == null) { + EXCEPTIONS = builder.build();//EXCEPTION构建成功 + } + //从EXCEPTIONS中找到异常类型所对应的错误代码,如果找到了,将错误代码相应给用户,如果找不到则给用户相应99999 + ResultCode resultCode = EXCEPTIONS.get(exception.getClass()); + if (resultCode != null) { + return new ResponseResult(resultCode); + } + //返回99999异常 + return new ResponseResult(CommonCode.SERVER_ERROR); + } + + static { + //定义异常类型所对应的错误代码 + builder.put(HttpMessageNotReadableException.class, CommonCode.INVALID_PARAM); + } + + @ExceptionHandler(CustomException.class)//捕获CustomException类型异常 + @ResponseBody + public ResponseResult customException(CustomException customException) { + customException.printStackTrace(); + //记录日志 + LOGGER.error("catch exception:{}", customException.getMessage()); + + ResultCode resultCode = customException.getResultCode(); + return new ResponseResult(resultCode); + }*/ +} diff --git a/src/main/java/com/msdw/tms/common/exception/RRException.java b/src/main/java/com/msdw/tms/common/exception/RRException.java new file mode 100644 index 0000000..ff05905 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/exception/RRException.java @@ -0,0 +1,61 @@ +/** + * Copyright (c) 2016-2019 人人开源 All rights reserved. + * + * https://www.renren.io + * + * 版权所有,侵权必究! + */ + +package com.msdw.tms.common.exception; + +/** + * 自定义异常 + * + * @author Mark sunlightcs@gmail.com + */ +public class RRException extends RuntimeException { + /*private static final long serialVersionUID = 1L; + + private String msg; + private int code = 500; + + public RRException(String msg) { + super(msg); + this.msg = msg; + } + + public RRException(String msg, Throwable e) { + super(msg, e); + this.msg = msg; + } + + public RRException(String msg, int code) { + super(msg); + this.msg = msg; + this.code = code; + } + + public RRException(String msg, int code, Throwable e) { + super(msg, e); + this.msg = msg; + this.code = code; + } + + public String getMsg() { + return msg; + } + + public void setMsg(String msg) { + this.msg = msg; + } + + public int getCode() { + return code; + } + + public void setCode(int code) { + this.code = code; + } + */ + +} diff --git a/src/main/java/com/msdw/tms/common/utils/AliyunOssUtil.java b/src/main/java/com/msdw/tms/common/utils/AliyunOssUtil.java new file mode 100644 index 0000000..05084d7 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/AliyunOssUtil.java @@ -0,0 +1,315 @@ +package com.msdw.tms.common.utils; + +import com.aliyun.oss.ClientException; +import com.aliyun.oss.OSSClient; +import com.aliyun.oss.OSSException; +import com.aliyun.oss.model.*; +import com.msdw.tms.config.AliyunOssConfig; +import org.springframework.web.multipart.MultipartFile; + +import javax.servlet.http.HttpServletResponse; +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.net.URLEncoder; +import java.text.SimpleDateFormat; +import java.util.Date; + + +public class AliyunOssUtil { + + private static String sdf = new SimpleDateFormat("yyyyMMdd").format(new Date()); + + /** + * 创建存储空间 + * + * @param client + * @param config + */ + public static void createBucket(OSSClient client, AliyunOssConfig config) { + // 判断存储空间是否存在,不存在,则创建 + if (!client.doesBucketExist(config.getBucketName())) { + CreateBucketRequest bucketRequest = new CreateBucketRequest(null); + // 设置仓库名称 + bucketRequest.setBucketName(config.getBucketName()); + // 设置仓库权限 + bucketRequest.setCannedACL(CannedAccessControlList.PublicRead); + // 创建仓库 + client.createBucket(bucketRequest); + } + } + + /** + * 上传文件 + * + * @param file 需要上传的文件 + * @param client + * @param config + * @return + * @throws OSSException + * @throws ClientException + * @throws IOException + */ + public static FilesResult uploadFiles(MultipartFile file, OSSClient client, AliyunOssConfig config) throws OSSException, ClientException, IOException { + FilesResult result = new FilesResult(); + // 创建存储空间 + createBucket(client, config); + // 获取文件名 + String fileUrl = file.getOriginalFilename(); + // 获取文件尾缀 + String ext = fileUrl.substring(fileUrl.lastIndexOf(".")); + // 获取文件类型 + String fileType = getFileType(ext); + //创建文件 + String folderName = ext.substring(ext.indexOf(".") + 1); + String folder = createFolder(client, config.getBucketName(), folderName); + // 组合储存路径 + String fileName = folder + "/" + sdf + "/" + System.currentTimeMillis() + ext; + long fileSize = file.getSize(); + + ObjectMetadata metadata = new ObjectMetadata(); + // 指定该Object被下载时的网页的缓存行为 + metadata.setCacheControl("no-cache"); + // 指定该Object下设置Header + metadata.setHeader("Pragma", "no-cache"); + // 指定该Object被下载时的内容编码格式 + metadata.setContentEncoding("utf-8"); + // 文件的MIME,定义文件的类型及网页编码,决定浏览器将以什么形式、什么编码读取文件。如果用户没有指定则根据Key或文件名的扩展名生成, + // 如果没有扩展名则填默认值application/octet-stream + metadata.setContentType(fileType); + // 指定该Object被下载时的名称(指示MINME用户代理如何显示附加的文件,打开或下载,及文件名称) + metadata.setContentDisposition("filename/filesize=" + file.getName() + "/" + fileSize + "Byte."); + // 上传文件 (上传文件流的形式) + client.putObject(config.getBucketName(), fileName, new ByteArrayInputStream(file.getBytes()), metadata); + result.setFileName(fileName); + result.setFileUrl(config.getSufferUrl() + fileName); + + if (null != client) + client.shutdown(); + + return result; + } + + /** + * 用户头像上传 + * + * @param file 需要上传的文件 + * @param client + * @param config + * @return + * @throws OSSException + * @throws ClientException + * @throws IOException + */ + public static FilesResult uploadUserAvatars(MultipartFile file, OSSClient client, AliyunOssConfig config) throws OSSException, ClientException, IOException { + FilesResult result = new FilesResult(); + String userAvatars = config.getUserAvatars(); + // 创建存储空间 + createBucket(client, config); + // 获取文件名 + String fileUrl = file.getOriginalFilename(); + // 获取文件尾缀 + String ext = fileUrl.substring(fileUrl.lastIndexOf(".")); + // 获取文件类型 + String fileType = getFileType(ext); + // 组合储存路径 + String fileName = userAvatars + "/" + sdf + "/" + System.currentTimeMillis() + ext; + long fileSize = file.getSize(); + + ObjectMetadata metadata = new ObjectMetadata(); + // 指定该Object被下载时的网页的缓存行为 + metadata.setCacheControl("no-cache"); + // 指定该Object下设置Header + metadata.setHeader("Pragma", "no-cache"); + // 指定该Object被下载时的内容编码格式 + metadata.setContentEncoding("utf-8"); + // 文件的MIME,定义文件的类型及网页编码,决定浏览器将以什么形式、什么编码读取文件。如果用户没有指定则根据Key或文件名的扩展名生成, + // 如果没有扩展名则填默认值application/octet-stream + metadata.setContentType(fileType); + // 指定该Object被下载时的名称(指示MINME用户代理如何显示附加的文件,打开或下载,及文件名称) + metadata.setContentDisposition("filename/filesize=" + file.getName() + "/" + fileSize + "Byte."); + // 上传文件 (上传文件流的形式) + client.putObject(config.getBucketName(), fileName, new ByteArrayInputStream(file.getBytes()), metadata); + result.setFileName(fileName); + result.setFileUrl(config.getSufferUrl() + fileName); + + if (null != client) + client.shutdown(); + + return result; + } + + /** + * 简单上传 + * + * @param client + * @param config + * @param objectName 表示上传文件到OSS时需要指定包含文件后缀在内的完整路径,例如abc/efg/123.jpg。 + * @param stream + * @return + */ + public static FilesResult uploadFiles(OSSClient client, AliyunOssConfig config, String objectName, ByteArrayInputStream stream) { + FilesResult result = new FilesResult(); + try { + PutObjectRequest putObjectRequest = new PutObjectRequest(config.getBucketName(), objectName, stream); + client.putObject(putObjectRequest); + result.setFileName(objectName); + result.setFileUrl(config.getSufferUrl() + objectName); + + } catch (Exception e) { + e.printStackTrace(); + throw e; + } finally { + if (null != client) + client.shutdown(); + } + return result; + } + + /** + * 文件下载 + * + * @param response + * @param client + * @param config + * @param objectName + */ + public static void downloadFiles(HttpServletResponse response, OSSClient client, AliyunOssConfig config, + String objectName) throws IOException { + + OSSObject ossObject = client.getObject(config.getBucketName(), objectName); + InputStream input = null; + OutputStream outputStream = null; + try { + // 获取文件名 + String fileName = objectName.substring(objectName.lastIndexOf("/") + 1); + //获取输入流 + input = ossObject.getObjectContent(); + // 获取OutputStream输出流 + outputStream = response.getOutputStream(); + byte[] buffer = new byte[input.available()]; + + response.setHeader("content-Type", "application/vnd.ms-excel"); + response.setHeader("Content-disposition", "attachment;filename=" + URLEncoder.encode(fileName, "UTF-8")); + response.flushBuffer(); + + for (int length = 0; (length = input.read(buffer)) > 0; ) { + outputStream.write(buffer, 0, length); + } + } catch (IOException e) { + e.printStackTrace(); + throw e; + } finally { + if (null != outputStream) { + try { + outputStream.close(); + } catch (IOException e) { + e.printStackTrace(); + } + } + if (null != input) { + try { + input.close(); + } catch (IOException e) { + e.printStackTrace(); + } + } + if (null != client) + client.shutdown(); + } + + } + + /** + * 删除存储空间buckName + * + * @param ossClient oss对象 + * @param bucketName 存储空间 + */ + public static void deleteBucket(OSSClient ossClient, String bucketName) { + ossClient.deleteBucket(bucketName); + } + + /** + * 创建模拟文件夹 + * + * @param ossClient oss连接 + * @param bucketName 存储空间 + * @param folder 模拟文件夹名如"qj_nanjing/" + * @return + */ + public static String createFolder(OSSClient ossClient, String bucketName, String folder) { + // 文件夹名 + final String keySuffixWithSlash = folder; + // 判断文件夹是否存在,不存在则创建 + if (!ossClient.doesObjectExist(bucketName, keySuffixWithSlash)) { + // 创建文件夹 + ossClient.putObject(bucketName, keySuffixWithSlash, new ByteArrayInputStream(new byte[0])); + // 得到文件夹名 + OSSObject object = ossClient.getObject(bucketName, keySuffixWithSlash); + String fileDir = object.getKey(); + return fileDir; + } + return keySuffixWithSlash; + } + + /** + * 根据key删除OSS服务器上的文件 + * + * @param ossClient oss连接 + * @param bucketName 存储空间 + * @param fileName 模拟文件夹名 如"qj_nanjing/" + */ + public static void deleteFile(OSSClient ossClient, String bucketName, String fileName) { + ossClient.deleteObject(bucketName, fileName); + } + + /** + * 获取文件类型 + * + * @param fileExtension 文件后缀 + * @return + */ + public static String getFileType(String fileExtension) { + if (".bmp".equalsIgnoreCase(fileExtension)) { + return "image/bmp"; + } + if (".gif".equalsIgnoreCase(fileExtension)) { + return "image/gif"; + } + if (".jpeg".equalsIgnoreCase(fileExtension) || ".jpg".equalsIgnoreCase(fileExtension) + || ".png".equalsIgnoreCase(fileExtension)) { + return "image/jpg"; + } + if (".html".equalsIgnoreCase(fileExtension)) { + return "text/html"; + } + if (".txt".equalsIgnoreCase(fileExtension)) { + return "text/plain"; + } + if (".vsd".equalsIgnoreCase(fileExtension)) { + return "application/vnd.visio"; + } + if (".ppt".equalsIgnoreCase(fileExtension) || ".pptx".equalsIgnoreCase(fileExtension)) { + return "application/vnd.ms-powerpoint"; + } + if (".doc".equalsIgnoreCase(fileExtension) || ".docx".equalsIgnoreCase(fileExtension)) { + return "application/msword"; + } + if (".xml".equalsIgnoreCase(fileExtension)) { + return "text/xml"; + } + if (".mp4".equalsIgnoreCase(fileExtension) || ".avi".equalsIgnoreCase(fileExtension)) { + return "video/mp4"; + } + if (".mpg".equalsIgnoreCase(fileExtension) || ".mpeg".equalsIgnoreCase(fileExtension)) { + return "video/mpeg"; + } + if ("pdf".equalsIgnoreCase(fileExtension)) { + return "application/pdf"; + } + return "application/octet-stream"; + } + +} diff --git a/src/main/java/com/msdw/tms/common/utils/BasePage.java b/src/main/java/com/msdw/tms/common/utils/BasePage.java new file mode 100644 index 0000000..fa1f604 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/BasePage.java @@ -0,0 +1,19 @@ +package com.msdw.tms.common.utils; + +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.entity + * @ClassName: BasePage + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/9 17:42 + * @UpdateDate: 2020/10/9 17:42 + * @Version: 1.0 + */ +@Data +public class BasePage { + private int pageSize = 10; + private int pageNum = 1; +} diff --git a/src/main/java/com/msdw/tms/common/utils/Constant.java b/src/main/java/com/msdw/tms/common/utils/Constant.java new file mode 100644 index 0000000..96214b0 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/Constant.java @@ -0,0 +1,489 @@ +/** + * Copyright (c) 2016-2019 人人开源 All rights reserved. + *

+ * https://www.renren.io + *

+ * 版权所有,侵权必究! + */ + +package com.msdw.tms.common.utils; + +/** + * 常量 + * + * @author Mark sunlightcs@gmail.com + */ +public class Constant { + /** + * 超级管理员ID + */ + public static final int SUPER_ADMIN = 1; + /** + * 管理员 + */ + public static final int ADMIN = 2; + /** + * 教师 + */ + public static final int TEACHER = 3; + /** + * 学生 + */ + public static final int STUDENT = 4; + /** + * 当前页码 + */ + public static final String PAGE = "page"; + /** + * 每页显示记录数 + */ + public static final String LIMIT = "limit"; + /** + * 排序字段 + */ + public static final String ORDER_FIELD = "sidx"; + /** + * 排序方式 + */ + public static final String ORDER = "order"; + /** + * 升序 + */ + public static final String ASC = "asc"; + /** + * 通过excel批量导入试题数据时起始行 + */ + public static final int ROW_INDEX = 2; + /** + * 通过excel批量导入试题数据时起始行 + */ + public static final int CELL_INDEX = 0; + /** + * 需要提取的样式所在的行号 + */ + public static final int STYLE_INDEX = 2; + /** + * 试题选项A + */ + public static final String A = "A"; + /** + * 试题选项B + */ + public static final String B = "B"; + /** + * 试题选项C + */ + public static final String C = "C"; + /** + * 试题选项D + */ + public static final String D = "D"; + /** + * 试题选项E + */ + public static final String E = "E"; + /** + * 试题选项F + */ + public static final String F = "F"; + /** + * excel模板信息表id + */ + public static final int XLSX_TEMPLATE_ID = 1; + /** + * 测评规则表id + */ + public static final int EVALUATION_RULES_ID = 1; + /** + * excel后缀 + */ + public static final String EXCEL_SUFFIX = ".xlsx"; + /** + * 测评总分 + */ + public static final int EVALUATION_SCORE = 100; + /** + * 及格分数 + */ + public static final int PASSING_SCORE = 60; + /** + * 测评是否通过 1、通过/0、未通过 + */ + public static final String EVALUATIONN_PASSED = "通过"; + public static final String EVALUATIONN_NOT_PASSED = "未通过"; + /** + * 菜单类型 + * + * @author chenshun + * @email sunlightcs@gmail.com + * @date 2016年11月15日 下午1:24:29 + */ + public enum MenuType { + /** + * 目录 + */ + CATALOG(0), + /** + * 菜单 + */ + MENU(1), + /** + * 按钮 + */ + BUTTON(2); + + private int value; + + MenuType(int value) { + this.value = value; + } + + public int getValue() { + return value; + } + } + + /** + * 定时任务状态 + * + * @author chenshun + * @email sunlightcs@gmail.com + * @date 2016年12月3日 上午12:07:22 + */ + public enum ScheduleStatus { + /** + * 正常 + */ + NORMAL(0), + /** + * 暂停 + */ + PAUSE(1); + + private int value; + + ScheduleStatus(int value) { + this.value = value; + } + + public int getValue() { + return value; + } + } + + /** + * 云服务商 + */ + public enum CloudService { + /** + * 七牛云 + */ + QINIU(1), + /** + * 阿里云 + */ + ALIYUN(2), + /** + * 腾讯云 + */ + QCLOUD(3); + + private int value; + + CloudService(int value) { + this.value = value; + } + + public int getValue() { + return value; + } + } + + /** + * 是否启用 + */ + public enum IsEnable { + /** + * 启用 + */ + ENABLE(1, "启用"), + /** + * 不启用 + */ + NOT_ENABLE(0, "不启用"); + + private Integer type; //类型 + private String desc; //描述 + + IsEnable(Integer type, String desc) { + this.type = type; + this.desc = desc; + } + + public Integer getType() { + return type; + } + + public void setType(Integer type) { + this.type = type; + } + + public String getDesc() { + return desc; + } + + public void setDesc(String desc) { + this.desc = desc; + } + } + + + /** + * 是否删除:0使用,1删除,默认0使用 + */ + public enum IsDel { + /** + * 删除 + */ + DEL(1, "已删除"), + /** + * 不删除 + */ + NOT_DEL(0, "不删除"); + + private Integer type; //类型 + private String desc; //描述 + + IsDel(Integer type, String desc) { + this.type = type; + this.desc = desc; + } + + public Integer getType() { + return type; + } + + public void setType(Integer type) { + this.type = type; + } + + public String getDesc() { + return desc; + } + + public void setDesc(String desc) { + this.desc = desc; + } + } + + /** + * 评测规则类型 + */ + public enum RulesType { + /** + * 随机 + */ + RANDOM(1, "随机"), + /** + * 自定义 + */ + CUSTOMIZE(0, "自定义"); + + private Integer type; //类型 + private String desc; //描述 + + RulesType(Integer type, String desc) { + this.type = type; + this.desc = desc; + } + + public Integer getType() { + return type; + } + + public void setType(Integer type) { + this.type = type; + } + + public String getDesc() { + return desc; + } + + public void setDesc(String desc) { + this.desc = desc; + } + } + + /** + * 试题类型 + */ + public enum QuestionType { + /** + * 单选题 + */ + SINGLE_CHOICE(1, "单选题"), + /** + * 多选题 + */ + MULTIPLE_CHOICE(2, "多选题"), + /** + * 判断题 + */ + TRUE_OR_FALSE(3, "判断题"); + + private Integer type; //类型 + private String desc; //描述 + + QuestionType(Integer type, String desc) { + this.type = type; + this.desc = desc; + } + + public Integer getType() { + return type; + } + + public void setType(Integer type) { + this.type = type; + } + + public String getDesc() { + return desc; + } + + public void setDesc(String desc) { + this.desc = desc; + } + + } + + /** + * 项目是否提交1:已提交/0:未提交 + */ + public enum Submit { + + SUBMIT(1, "已提交"), + NOT_SUBMIT(0, "未提交"); + + + private Integer type; + private String desc; + + Submit(Integer type, String desc) { + this.type = type; + this.desc = desc; + } + + public Integer getType() { + return type; + } + + public void setType(Integer type) { + this.type = type; + } + + public String getDesc() { + return desc; + } + + public void setDesc(String desc) { + this.desc = desc; + } + } + + public enum Ascription { + + SYSTEM(1, "系统内置"), + USER(0, "用户自建"); + + private Integer type; + private String desc; + + Ascription(Integer type, String desc) { + this.type = type; + this.desc = desc; + } + + public Integer getType() { + return type; + } + + public void setType(Integer type) { + this.type = type; + } + + public String getDesc() { + return desc; + } + + public void setDesc(String desc) { + this.desc = desc; + } + } + + /** + * 当前题目答案是否正确 1、正确/0、错误 + */ + public enum QuestionIsTure { + + TRUE(1, "正确"), + FALSE(0, "错误"); + + private Integer type; + private String desc; + + QuestionIsTure(Integer type, String desc) { + this.type = type; + this.desc = desc; + } + + public Integer getType() { + return type; + } + + public void setType(Integer type) { + this.type = type; + } + + public String getDesc() { + return desc; + } + + public void setDesc(String desc) { + this.desc = desc; + } + } + + /** + * 当前题目状态 1、已作/0、未作 + */ + public enum QuestionStatus { + + MADE(1, "已作"), + NOT_MADE(0, "未作"); + + private Integer type; + private String desc; + + QuestionStatus(Integer type, String desc) { + this.type = type; + this.desc = desc; + } + + public Integer getType() { + return type; + } + + public void setType(Integer type) { + this.type = type; + } + + public String getDesc() { + return desc; + } + + public void setDesc(String desc) { + this.desc = desc; + } + } +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/common/utils/DownloadUtils.java b/src/main/java/com/msdw/tms/common/utils/DownloadUtils.java new file mode 100644 index 0000000..57c0099 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/DownloadUtils.java @@ -0,0 +1,19 @@ +package com.msdw.tms.common.utils; + +import javax.servlet.ServletOutputStream; +import javax.servlet.http.HttpServletResponse; +import java.io.ByteArrayOutputStream; +import java.io.IOException; + +public class DownloadUtils { + public void download(ByteArrayOutputStream byteArrayOutputStream, HttpServletResponse response, String returnName) throws IOException { + response.setContentType("application/octet-stream"); + returnName = response.encodeURL(new String(returnName.getBytes(), "iso8859-1")); //保存的文件名,必须和页面编码一致,否则乱码 + response.addHeader("content-disposition", "attachment;filename=" + returnName); + response.setContentLength(byteArrayOutputStream.size()); + ServletOutputStream outputstream = response.getOutputStream(); //取得输出流 + byteArrayOutputStream.writeTo(outputstream); //写到输出流 + byteArrayOutputStream.close(); //关闭 + outputstream.flush(); //刷数据 + } +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/common/utils/FilesResult.java b/src/main/java/com/msdw/tms/common/utils/FilesResult.java new file mode 100644 index 0000000..aa36a77 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/FilesResult.java @@ -0,0 +1,15 @@ +package com.msdw.tms.common.utils; + +import lombok.Data; +import lombok.ToString; + +@Data +@ToString +public class FilesResult { + // 文件名 + private String fileName; + // 文件在储存空间的路径 + private String fileUrl; + // 上传状态 + private String status; +} diff --git a/src/main/java/com/msdw/tms/common/utils/MoneyConversionUtil.java b/src/main/java/com/msdw/tms/common/utils/MoneyConversionUtil.java new file mode 100644 index 0000000..337b341 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/MoneyConversionUtil.java @@ -0,0 +1,27 @@ +package com.msdw.tms.common.utils; + +import java.math.BigDecimal; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common.utils + * @ClassName: MoneyConversionUtil + * @Description: 钱换算 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/2 19:34 + * @UpdateDate: 2020/11/2 19:34 + * @Version: 1.0 + */ +public class MoneyConversionUtil { + + + public static BigDecimal moneyConversion(Double money) { + BigDecimal bigDecimal1 = new BigDecimal(Double.toString(money)); + BigDecimal bigDecimal2 = new BigDecimal(10000); + double val = bigDecimal1.multiply(bigDecimal2).doubleValue(); + BigDecimal bigDecimal = new BigDecimal(val); + System.out.println(bigDecimal.toString()); + + return bigDecimal; + } +} diff --git a/src/main/java/com/msdw/tms/common/utils/PageUtil.java b/src/main/java/com/msdw/tms/common/utils/PageUtil.java new file mode 100644 index 0000000..63be79b --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/PageUtil.java @@ -0,0 +1,22 @@ +package com.msdw.tms.common.utils; + +import com.baomidou.mybatisplus.annotation.TableField; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common.utils + * @ClassName: PageUtil + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 10:49 + * @UpdateDate: 2020/10/23 10:49 + * @Version: 1.0 + */ +@Data +public class PageUtil { + @TableField(exist = false) + private int pageSize = 10; + @TableField(exist = false) + private int pageNum = 0; +} diff --git a/src/main/java/com/msdw/tms/common/utils/PageUtils.java b/src/main/java/com/msdw/tms/common/utils/PageUtils.java new file mode 100644 index 0000000..aa7221b --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/PageUtils.java @@ -0,0 +1,112 @@ +/** + * Copyright (c) 2016-2019 人人开源 All rights reserved. + *

+ * https://www.renren.io + *

+ * 版权所有,侵权必究! + */ + +package com.msdw.tms.common.utils; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import lombok.Data; + +import java.io.Serializable; +import java.util.List; + +/** + * 分页工具类 + * + * @author Mark sunlightcs@gmail.com + */ +@Data +public class PageUtils implements Serializable { + private static final long serialVersionUID = 1L; + /** + * 总记录数 + */ + private int totalCount; + /** + * 每页记录数 + */ + private int pageSize; + /** + * 总页数 + */ + private int totalPage; + /** + * 当前页数 + */ + private int currPage; + /** + * 列表数据 + */ + private List list; + + /** + * 分页 + * @param list 列表数据 + * @param totalCount 总记录数 + * @param pageSize 每页记录数 + * @param currPage 当前页数 + */ + public PageUtils(List list, int totalCount, int pageSize, int currPage) { + this.list = list; + this.totalCount = totalCount; + this.pageSize = pageSize; + this.currPage = currPage; + this.totalPage = (int) Math.ceil((double) totalCount / pageSize); + } + + /** + * 分页 + */ + public PageUtils(IPage page) { + this.list = page.getRecords(); + this.totalCount = (int) page.getTotal(); + this.pageSize = (int) page.getSize(); + this.currPage = (int) page.getCurrent(); + this.totalPage = (int) page.getPages(); + } + + public int getTotalCount() { + return totalCount; + } + + public void setTotalCount(int totalCount) { + this.totalCount = totalCount; + } + + public int getPageSize() { + return pageSize; + } + + public void setPageSize(int pageSize) { + this.pageSize = pageSize; + } + + public int getTotalPage() { + return totalPage; + } + + public void setTotalPage(int totalPage) { + this.totalPage = totalPage; + } + + public int getCurrPage() { + return currPage; + } + + public void setCurrPage(int currPage) { + this.currPage = currPage; + } + + public List getList() { + return list; + } + + public void setList(List list) { + this.list = list; + } + +} diff --git a/src/main/java/com/msdw/tms/common/utils/Query.java b/src/main/java/com/msdw/tms/common/utils/Query.java new file mode 100644 index 0000000..f94a521 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/Query.java @@ -0,0 +1,94 @@ +/** + * Copyright (c) 2016-2019 人人开源 All rights reserved. + *

+ * https://www.renren.io + *

+ * 版权所有,侵权必究! + */ + +package com.msdw.tms.common.utils; + +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.core.metadata.OrderItem; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.msdw.tms.common.xss.SQLFilter; +import org.apache.commons.lang.StringUtils; + +import java.util.Map; + +/** + * 查询参数 + * + * @author Mark sunlightcs@gmail.com + */ +public class Query { + + public IPage getPage(Map params) { + return this.getPage(params, null, false); + } + + public IPage getPage(Map params, String defaultOrderField, boolean isAsc) { + //分页参数 + long curPage = 1; + long limit = 10; + + if (params.get(Constant.PAGE) != null) { + curPage = Long.parseLong((String) params.get(Constant.PAGE)); + } + if (params.get(Constant.LIMIT) != null) { + limit = Long.parseLong((String) params.get(Constant.LIMIT)); + } + + //分页对象 + Page page = new Page<>(curPage, limit); + + //分页参数 + params.put(Constant.PAGE, page); + + //排序字段 + //防止SQL注入(因为sidx、order是通过拼接SQL实现排序的,会有SQL注入风险) + String orderField = SQLFilter.sqlInject((String) params.get(Constant.ORDER_FIELD)); + String order = (String) params.get(Constant.ORDER); + + //前端字段排序 + if (StringUtils.isNotEmpty(orderField) && StringUtils.isNotEmpty(order)) { + if (Constant.ASC.equalsIgnoreCase(order)) { + return page.addOrder(OrderItem.asc(orderField)); + } else { + return page.addOrder(OrderItem.desc(orderField)); + } + } + + //没有排序字段,则不排序 + if (StringUtils.isBlank(defaultOrderField)) { + return page; + } + + //默认排序 + if (isAsc) { + page.addOrder(OrderItem.asc(defaultOrderField)); + } else { + page.addOrder(OrderItem.desc(defaultOrderField)); + } + + return page; + } + + public IPage getPage(Integer pageNo, Integer size) { + //分页参数 + long curPage = 1; + long limit = 10; + + if (pageNo != null) { + curPage = Long.parseLong(pageNo.toString()); + } + if (size != null) { + limit = Long.parseLong(size.toString()); + } + + //分页对象 + Page page = new Page<>(curPage, limit); + + return page; + } +} diff --git a/src/main/java/com/msdw/tms/common/utils/R.java b/src/main/java/com/msdw/tms/common/utils/R.java new file mode 100644 index 0000000..ea7caf9 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/R.java @@ -0,0 +1,63 @@ +/** + * Copyright (c) 2016-2019 人人开源 All rights reserved. + *

+ * https://www.renren.io + *

+ * 版权所有,侵权必究! + */ + +package com.msdw.tms.common.utils; + +import java.util.HashMap; +import java.util.Map; + +/** + * 返回数据 + * + * @author Mark sunlightcs@gmail.com + */ +public class R extends HashMap { + private static final long serialVersionUID = 1L; + + public R() { + put("status", 200); + put("errmessage", "success"); + } + + public static R error() { + return error(500, "未知异常,请联系管理员"); + } + + public static R error(String msg) { + return error(500, msg); + } + + public static R error(int code, String msg) { + R r = new R(); + r.put("status", code); + r.put("errmessage", msg); + return r; + } + + public static R ok(String msg) { + R r = new R(); + r.put("errmessage", msg); + return r; + } + + public static R ok(Map map) { + R r = new R(); + r.putAll(map); + return r; + } + + public static R ok() { + return new R(); + } + + public R put(String key, Object value) { + super.put(key, value); + return this; + } + +} diff --git a/src/main/java/com/msdw/tms/common/utils/RedisConfig.java b/src/main/java/com/msdw/tms/common/utils/RedisConfig.java new file mode 100644 index 0000000..950399b --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/RedisConfig.java @@ -0,0 +1,41 @@ +package com.msdw.tms.common.utils; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.redis.core.RedisTemplate; +import org.springframework.stereotype.Component; + +import javax.annotation.PostConstruct; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common.utils + * @ClassName: RedisConfig + * @Description: redis配置类, 用于更改RedisTemplate来创建工具类 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/13 9:35 + * @UpdateDate: 2020/11/13 9:35 + * @Version: 1.0 + */ +@Component +public class RedisConfig { + @Autowired + private RedisTemplate redisTemplate; + public static RedisTemplate redis; + + + @PostConstruct + public void getRedisTemplate() { + redis = this.redisTemplate; + } + + /** + * 读取缓存 + * + * @param key + * @return + */ + public Object get(final String key) { + return redis.opsForValue().get(key); + } + +} diff --git a/src/main/java/com/msdw/tms/common/utils/RestUtil.java b/src/main/java/com/msdw/tms/common/utils/RestUtil.java new file mode 100644 index 0000000..e864c30 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/RestUtil.java @@ -0,0 +1,186 @@ +package com.msdw.tms.common.utils; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.InputStreamReader; +import java.io.PrintWriter; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common.utils + * @ClassName: RestUtil + * @Description: java调用接口方式 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/13 11:26 + * @UpdateDate: 2020/11/13 11:26 + * @Version: 1.0 + */ +public class RestUtil { + + + public final static String postUrlBysave = "http://localhost:8080/sichuan/chuanda/userAssets/save"; + //远程调用接口 + + + /** + * 发送GET请求 + * + * @param url 目的地址 + * @param parameters 请求参数,Map类型。 + * @return 远程响应结果 + */ + public static String sendGet(String url, Map parameters) { + String result = ""; + BufferedReader in = null;// 读取响应输入流 + StringBuffer sb = new StringBuffer();// 存储参数 + String params = "";// 编码之后的参数 + try { + // 编码请求参数 + if (parameters.size() == 1) { + for (String name : parameters.keySet()) { + sb.append(name).append("=").append( + java.net.URLEncoder.encode(parameters.get(name), + "UTF-8")); + } + params = sb.toString(); + } else { + for (String name : parameters.keySet()) { + sb.append(name).append("=").append( + java.net.URLEncoder.encode(parameters.get(name), + "UTF-8")).append("&"); + } + String temp_params = sb.toString(); + params = temp_params.substring(0, temp_params.length() - 1); + } + String full_url = url + "?" + params; + System.out.println(full_url); + // 创建URL对象 + java.net.URL connURL = new java.net.URL(full_url); + // 打开URL连接 + java.net.HttpURLConnection httpConn = (java.net.HttpURLConnection) connURL + .openConnection(); + // 设置通用属性 + httpConn.setRequestProperty("Accept", "*/*"); + httpConn.setRequestProperty("Connection", "Keep-Alive"); + httpConn.setRequestProperty("User-Agent", + "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.1)"); + // 建立实际的连接 + httpConn.connect(); + // 响应头部获取 + Map> headers = httpConn.getHeaderFields(); + // 遍历所有的响应头字段 + for (String key : headers.keySet()) { + System.out.println(key + "\t:\t" + headers.get(key)); + } + // 定义BufferedReader输入流来读取URL的响应,并设置编码方式 + in = new BufferedReader(new InputStreamReader(httpConn + .getInputStream(), "UTF-8")); + String line; + // 读取返回的内容 + while ((line = in.readLine()) != null) { + result += line; + } + } catch (Exception e) { + e.printStackTrace(); + } finally { + try { + if (in != null) { + in.close(); + } + } catch (IOException ex) { + ex.printStackTrace(); + } + } + return result; + } + + /** + * 发送POST请求 + * + * @param url 目的地址 + * @param parameters 请求参数,Map类型。 + * @return 远程响应结果 + */ + public static String sendPost(String url, Map parameters) { + String result = "";// 返回的结果 + BufferedReader in = null;// 读取响应输入流 + PrintWriter out = null; + StringBuffer sb = new StringBuffer();// 处理请求参数 + String params = "";// 编码之后的参数 + try { + // 编码请求参数 + if (parameters.size() == 1) { + for (String name : parameters.keySet()) { + sb.append(name).append("=").append( + java.net.URLEncoder.encode(parameters.get(name), + "UTF-8")); + } + params = sb.toString(); + } else { + for (String name : parameters.keySet()) { + sb.append(name).append("=").append( + java.net.URLEncoder.encode(parameters.get(name), + "UTF-8")).append("&"); + } + String temp_params = sb.toString(); + params = temp_params.substring(0, temp_params.length() - 1); + } + // 创建URL对象 + java.net.URL connURL = new java.net.URL(url); + // 打开URL连接 + java.net.HttpURLConnection httpConn = (java.net.HttpURLConnection) connURL + .openConnection(); + // 设置通用属性 + httpConn.setRequestProperty("Accept", "*/*"); + httpConn.setRequestProperty("Connection", "Keep-Alive"); + httpConn.setRequestProperty("User-Agent", + "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.1)"); + // 设置POST方式 + httpConn.setDoInput(true); + httpConn.setDoOutput(true); + // 获取HttpURLConnection对象对应的输出流 + out = new PrintWriter(httpConn.getOutputStream()); + // 发送请求参数 + out.write(params); + // flush输出流的缓冲 + out.flush(); + // 定义BufferedReader输入流来读取URL的响应,设置编码方式 + in = new BufferedReader(new InputStreamReader(httpConn + .getInputStream(), "UTF-8")); + String line; + // 读取返回的内容 + while ((line = in.readLine()) != null) { + result += line; + } + } catch (Exception e) { + e.printStackTrace(); + } finally { + try { + if (out != null) { + out.close(); + } + if (in != null) { + in.close(); + } + } catch (IOException ex) { + ex.printStackTrace(); + } + } + return result; + } + + /** + * 主函数,测试请求 + * + * @param args + */ + public static void main(String[] args) { + Map parameters = new HashMap<>(); + parameters.put("userId", "10001"); + String result = sendPost("http://localhost:8080/sichuan/chuanda/userAssets/save", parameters); + System.out.println(result); + } +} diff --git a/src/main/java/com/msdw/tms/common/utils/TokenUtil.java b/src/main/java/com/msdw/tms/common/utils/TokenUtil.java new file mode 100644 index 0000000..dd3ed80 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/TokenUtil.java @@ -0,0 +1,84 @@ +package com.msdw.tms.common.utils; + +import sun.misc.BASE64Encoder; + +import java.security.MessageDigest; +import java.security.NoSuchAlgorithmException; +import java.util.Random; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common.utils + * @ClassName: TokenUtil + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/22 15:10 + * @UpdateDate: 2020/10/22 15:10 + * @Version: 1.0 + */ +public class TokenUtil { + + + private static int userIdNum = 0; + + public static int getUserId() { + + /*RedisTemplate redisTemplate = RedisConfig.redis; + *//*HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest(); + //获取请求头中的token + String token = request.getHeader("token");*//* + + String token = "UT_tfHcVAEB2I4pUZU0PgryDg=="; + Map parameters = new HashMap(); + + //判断redis中是否有有这个token + String redisToken = redisTemplate.opsForValue().get(token); + int userId = 0; + if (redisToken == null) { + //redisToken为null表示reids中没有这个token,说明从专家通道进入 + userId = getUserIdNum(); + parameters.put("userId", String.valueOf(userId)); + + RestUtil.sendPost(RestUtil.postUrlBysave, parameters); + return userId; + } else { + //从学生通道进入的有token + parameters.put("userId", redisToken); + RestUtil.sendPost(RestUtil.postUrlBysave, parameters); + return Integer.parseInt(redisToken); + }*/ + return 1; + + } + + public static int getUserIdNum() { + return userIdNum += 1; + } + + + /** + * 生成Token + * Token:Nv6RRuGEVvmGjB+jimI/gw== + * + * @return + */ + public static String makeToken() { + String token = (System.currentTimeMillis() + new Random().nextInt(999999999)) + ""; + //数据指纹 128位长 16个字节 md5 + try { + MessageDigest md = MessageDigest.getInstance("md5"); + byte md5[] = md.digest(token.getBytes()); + //base64编码--任意二进制编码明文字符 adfsdfsdfsf + BASE64Encoder encoder = new BASE64Encoder(); + return encoder.encode(md5); + } catch (NoSuchAlgorithmException e) { + throw new RuntimeException(e); + } + } + + public static void main(String[] args) { + System.out.println(makeToken()); + } + + +} diff --git a/src/main/java/com/msdw/tms/common/utils/date/DateUtils.java b/src/main/java/com/msdw/tms/common/utils/date/DateUtils.java new file mode 100644 index 0000000..12b4774 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/date/DateUtils.java @@ -0,0 +1,99 @@ +package com.msdw.tms.common.utils.date; + +import java.text.ParseException; +import java.text.SimpleDateFormat; +import java.time.Instant; +import java.time.LocalDateTime; +import java.time.ZoneId; +import java.time.format.DateTimeFormatter; +import java.util.Calendar; +import java.util.Date; + +public class DateUtils { + + public static String handleTime(Date date) { + Instant instant = date.toInstant(); + ZoneId zoneId = ZoneId.systemDefault(); + LocalDateTime localDateTime = instant.atZone(zoneId).toLocalDateTime(); + return localDateTime.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")); // 当前日期和时间 + } + + + public static String formatTime(LocalDateTime time) { + String timeStr1 = time.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")); + System.out.println("当前时间为:" + timeStr1); + return timeStr1; + } + + /** + * 时间类型转换 + * + * @param time + * @return + */ + public static String dateForByStr(String time) { + try { + SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); + Date date = sdf.parse(time); + //获取String类型的时间 + String createdate = sdf.format(date); + return createdate; + } catch (ParseException e) { + e.printStackTrace(); + return null; + } + + } + + + /** + * 获取当前时间 + * + * @return + */ + /*public static String getNowTime(Integer userId) { + + String getDate = template.opsForValue().get("date:userId_" + userId); + SimpleDateFormat tempDate = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); + String datetime = tempDate.format(new java.util.Date()); + //也可以用这个 + String datetimes = tempDate.format(new Date(System.currentTimeMillis())); + return getDate; + }*/ + + /*public static String getNowTime() { + SimpleDateFormat tempDate = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); + String datetime = tempDate.format(new java.util.Date()); + //也可以用这个 + String datetimes = tempDate.format(new Date(System.currentTimeMillis())); + return datetime; + }*/ + + /** + * 获取当前时间往后推几个月 + * + * @param inputDate + * @param number + * @return + */ + public static String getAfterMonth(String inputDate, int number) { + Calendar c = Calendar.getInstance();//获得一个日历的实例 + SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); + Date date = null; + try { + date = sdf.parse(inputDate);//初始日期 + } catch (Exception e) { + + } + c.setTime(date);//设置日历时间 + c.add(Calendar.MONTH, number);//在日历的月份上增加6个月 + String strDate = sdf.format(c.getTime());//的到你想要得6个月后的日期 + return strDate; + } + + public static void main(String[] args) { + String date = getAfterMonth("2020-09-10", 3); + System.out.println(date); + } + +} diff --git a/src/main/java/com/msdw/tms/common/utils/photo/photoUrl.java b/src/main/java/com/msdw/tms/common/utils/photo/photoUrl.java new file mode 100644 index 0000000..cfb87b7 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/photo/photoUrl.java @@ -0,0 +1,15 @@ +package com.msdw.tms.common.utils.photo; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common.utils.photo + * @ClassName: photoUrl + * @Description: 图片 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/5 15:35 + * @UpdateDate: 2020/11/5 15:35 + * @Version: 1.0 + */ +public class photoUrl { + private static String getPhotoUrl = ""; +} diff --git a/src/main/java/com/msdw/tms/common/utils/poi/EasyExcelUtil.java b/src/main/java/com/msdw/tms/common/utils/poi/EasyExcelUtil.java new file mode 100644 index 0000000..6f41ba7 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/poi/EasyExcelUtil.java @@ -0,0 +1,28 @@ +package com.msdw.tms.common.utils.poi; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common.utils.poi + * @ClassName: EasyExcelUtil + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 12:40 + * @UpdateDate: 2020/10/23 12:40 + * @Version: 1.0 + */ + +import com.alibaba.excel.EasyExcel; + +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; +import java.util.List; + +public class EasyExcelUtil { + public static void download(HttpServletResponse response, Class head, List list,String filename) throws IOException { + response.setContentType("application/vnd.ms-excel"); + response.setCharacterEncoding("utf-8"); + response.setHeader("Content-disposition", "attachment;filename="+filename+".xlsx"); + EasyExcel.write(response.getOutputStream(), head).sheet(filename).doWrite(list); + } +} + diff --git a/src/main/java/com/msdw/tms/common/utils/poi/ExcelAttribute.java b/src/main/java/com/msdw/tms/common/utils/poi/ExcelAttribute.java new file mode 100644 index 0000000..31b70b9 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/poi/ExcelAttribute.java @@ -0,0 +1,25 @@ +package com.msdw.tms.common.utils.poi; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +@Retention(RetentionPolicy.RUNTIME) +@Target(ElementType.FIELD) +public @interface ExcelAttribute { + /** + * 对应的列名称 + */ + String name() default ""; + + /** + * excel列的索引 + */ + int sort(); + + /** + * 字段类型对应的格式 + */ + String format() default ""; +} diff --git a/src/main/java/com/msdw/tms/common/utils/poi/ExcelExportUtil.java b/src/main/java/com/msdw/tms/common/utils/poi/ExcelExportUtil.java new file mode 100644 index 0000000..ffe4899 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/poi/ExcelExportUtil.java @@ -0,0 +1,84 @@ +package com.msdw.tms.common.utils.poi; + +import lombok.Data; +import org.apache.poi.ss.usermodel.*; +import org.apache.poi.xssf.usermodel.XSSFWorkbook; + +import javax.servlet.http.HttpServletResponse; +import java.io.InputStream; +import java.lang.reflect.Field; +import java.net.URLEncoder; +import java.util.List; +import java.util.concurrent.atomic.AtomicInteger; + +/** + * 导出Excel工具类 + * 基于模板打印的方式导出: + */ +@Data +public class ExcelExportUtil { + + private int rowIndex; //写入数据的起始行 + private int styleIndex; //需要提取的样式所在的行号 + private Class clazz; //对象的字节码 + private Field fields[]; //对象中的所有属性 + + public ExcelExportUtil(Class clazz, int rowIndex, int styleIndex) { + this.clazz = clazz; + this.rowIndex = rowIndex; + this.styleIndex = styleIndex; + fields = clazz.getDeclaredFields(); + } + + /** + * 基于注解导出 + * 参数: + * response: + * InputStream:模板的输入流 + * objs:数据 + * fileName:生成的文件名 + */ + public void export(HttpServletResponse response, InputStream is, List objs, String fileName) throws Exception { + + //1.根据模板创建工作簿 + XSSFWorkbook workbook = new XSSFWorkbook(is); + //2.读取工作表 + Sheet sheet = workbook.getSheetAt(0); + //3.提取公共的样式 + CellStyle[] styles = getTemplateStyles(sheet.getRow(styleIndex)); + //4.根据数据创建每一行和每一个单元格的数据2 + AtomicInteger datasAi = new AtomicInteger(rowIndex); //数字 + for (T t : objs) { + //datasAi.getAndIncrement() :获取数字,并++ i++ + Row row = sheet.createRow(datasAi.getAndIncrement()); + for (int i = 0; i < styles.length; i++) { + Cell cell = row.createCell(i); + cell.setCellStyle(styles[i]); + for (Field field : fields) { + if (field.isAnnotationPresent(ExcelAttribute.class)) { + field.setAccessible(true); + ExcelAttribute ea = field.getAnnotation(ExcelAttribute.class); + if (i == ea.sort()) { + if (field.get(t) != null) { + cell.setCellValue(field.get(t).toString()); + } + } + } + } + } + } + fileName = URLEncoder.encode(fileName, "UTF-8"); + response.setContentType("application/octet-stream"); + response.setHeader("content-disposition", "attachment;filename=" + new String(fileName.getBytes("ISO8859-1"))); + response.setHeader("filename", fileName); + workbook.write(response.getOutputStream()); + } + + CellStyle[] getTemplateStyles(Row row) { + CellStyle[] styles = new CellStyle[row.getLastCellNum()]; + for (int i = 0; i < row.getLastCellNum(); i++) { + styles[i] = row.getCell(i).getCellStyle(); + } + return styles; + } +} diff --git a/src/main/java/com/msdw/tms/common/utils/poi/ExcelImportUtil.java b/src/main/java/com/msdw/tms/common/utils/poi/ExcelImportUtil.java new file mode 100644 index 0000000..73b6c98 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/poi/ExcelImportUtil.java @@ -0,0 +1,123 @@ +package com.msdw.tms.common.utils.poi; + +import org.apache.poi.ss.usermodel.Cell; +import org.apache.poi.ss.usermodel.DateUtil; +import org.apache.poi.ss.usermodel.Row; +import org.apache.poi.ss.usermodel.Sheet; +import org.apache.poi.xssf.usermodel.XSSFWorkbook; + +import java.io.InputStream; +import java.lang.reflect.Field; +import java.math.BigDecimal; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + + +public class ExcelImportUtil { + + private Class clazz; + private Field fields[]; + + public ExcelImportUtil(Class clazz) { + this.clazz = clazz; + fields = clazz.getDeclaredFields(); + } + + /** + * 基于注解读取excel + */ + public List readExcel(InputStream is, int rowIndex, int cellIndex) { + List list = new ArrayList<>(); + T entity = null; + try { + XSSFWorkbook workbook = new XSSFWorkbook(is); + Sheet sheet = workbook.getSheetAt(0); + // 不准确 + // int rowLength = sheet.getLastRowNum(); + // System.out.println(sheet.getLastRowNum()); + for (int rowNum = rowIndex; rowNum <= sheet.getLastRowNum(); rowNum++) { + Row row = sheet.getRow(rowNum); + entity = (T) clazz.newInstance(); + // System.out.println(row.getLastCellNum()); + for (int j = cellIndex; j < row.getLastCellNum(); j++) { + Cell cell = row.getCell(j); + for (Field field : fields) { + if (field.isAnnotationPresent(ExcelAttribute.class)) { + field.setAccessible(true); + ExcelAttribute ea = field.getAnnotation(ExcelAttribute.class); + if (j == ea.sort()) { + field.set(entity, covertAttrType(field, cell)); + } + } + } + } + Field field = entity.getClass().getDeclaredField("index"); + field.setAccessible(true); + field.set(entity, rowNum + 1); + list.add(entity); + } + } catch (Exception e) { + e.printStackTrace(); + } + return list; + } + + /** + * 类型转换 将cell 单元格格式转为 字段类型 + */ + private Object covertAttrType(Field field, Cell cell) throws Exception { + String fieldType = field.getType().getSimpleName(); + if ("String".equals(fieldType)) { + return getValue(cell); + } else if ("Date".equals(fieldType)) { + return new SimpleDateFormat("yyyy-MM-dd hh:mm:ss").parse(getValue(cell)); + } else if ("int".equals(fieldType) || "Integer".equals(fieldType)) { + return Integer.parseInt(getValue(cell)); + } else if ("double".equals(fieldType) || "Double".equals(fieldType)) { + return Double.parseDouble(getValue(cell)); + } else { + return null; + } + } + + + /** + * 格式转为String + * + * @param cell + * @return + */ + public String getValue(Cell cell) { + if (cell == null) { + return ""; + } + switch (cell.getCellType()) { + case STRING: + return cell.getRichStringCellValue().getString().trim(); + case NUMERIC: + if (DateUtil.isCellDateFormatted(cell)) { + Date dt = DateUtil.getJavaDate(cell.getNumericCellValue()); + return new SimpleDateFormat("yyyy-MM-dd hh:mm:ss").format(dt); + } else { + // 防止数值变成科学计数法 + String strCell = ""; + Double num = cell.getNumericCellValue(); + BigDecimal bd = new BigDecimal(num.toString()); + if (bd != null) { + strCell = bd.toPlainString(); + } + // 去除 浮点型 自动加的 .0 + if (strCell.endsWith(".0")) { + strCell = strCell.substring(0, strCell.indexOf(".")); + } + return strCell; + } + case BOOLEAN: + return String.valueOf(cell.getBooleanCellValue()); + default: + return ""; + } + } +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/common/utils/py/PyUtil.java b/src/main/java/com/msdw/tms/common/utils/py/PyUtil.java new file mode 100644 index 0000000..2c3820f --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/py/PyUtil.java @@ -0,0 +1,211 @@ +package com.msdw.tms.common.utils.py; + +import java.io.BufferedReader; +import java.io.IOException; +import java.io.InputStreamReader; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common.utils.py + * @ClassName: python + * @Description: Java调用Python脚本辅助类 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/7 13:15 + * @UpdateDate: 2020/10/7 13:15 + * @Version: 1.0 + */ +public class PyUtil { + + /*本地测试环境*/ + + //这里指的是python环境的地址 + /*public static String path = "python"; + //生成图片 + public static String pyPathByPhoto = "C:\\Users\\Hello\\Desktop\\python\\photo.py"; + //正态Var + public static String pyPathByVar = "C:\\Users\\Hello\\Desktop\\python\\getVar.py"; + //历史模拟法 + public static String historyPath = "C:\\Users\\Hello\\Desktop\\python\\getVarByHistory.py"; + //对冲前后生成图片 + public static String beforeAndAfter = "C:\\Users\\Hello\\Desktop\\python\\beforeAndAfterByHedging.py";*/ + /** + * 线上测试环境 + */ + //这里指的是python环境的地址 + public static String path = "python"; + //生成图片 + public static String pyPathByPhoto = "/usr/local/sichuan/py/photo.py"; + //正态Var + public static String pyPathByVar = "/usr/local/sichuan/py/getVar.py"; + //历史模拟法 + public static String historyPath = "/usr/local/sichuan/py/getVarByHistory.py"; + //对冲前后生成图片 + public static String beforeAndAfter = "/usr/local/sichuan/py/beforeAndAfterByHedging.py"; + + + /** + * 动态传参调用Python脚本生成图片 + * + * @param pyPath + * @param path + * @param params + * @throws IOException + */ + public static String getCodeByPhoto(String pyPath, String path, String... params) throws IOException { + try { + + String[] args = new String[2 + params.length]; + args[0] = pyPath; + args[1] = path; + for (int x = 0; x < args.length; x++) { + if (x > 1) { + args[x] = params[x - 2]; + } + } + //执行py文件 + Process proc = Runtime.getRuntime().exec(args); + BufferedReader in = new BufferedReader(new InputStreamReader(proc.getInputStream())); + String line = null; + String url = ""; + + + while ((line = in.readLine()) != null) { + url += line + ","; + System.out.println("获取结果:" + line); + } + + in.close(); + proc.waitFor(); + + + return url; + + //System.out.println("拼接后:" + url); + + + } catch (Exception e) { + e.printStackTrace(); + + } + return null; + + } + + + /** + * 动态传参调用Python脚本:正态Var + * + * @param pyPath + * @param path + * @param params + * @throws IOException + */ + public static String getVarCode(String pyPath, String path, String... params) throws IOException { + try { + + String[] args = new String[2 + params.length]; + args[0] = pyPath; + args[1] = path; + for (int x = 0; x < args.length; x++) { + if (x > 1) { + args[x] = params[x - 2]; + } + } + //执行py文件 + Process proc = Runtime.getRuntime().exec(args); + BufferedReader in = new BufferedReader(new InputStreamReader(proc.getInputStream())); + String line = null; + + String str = ""; + + while ((line = in.readLine()) != null) { + System.out.println("循环中输出:" + line); + str += line + ","; + //System.out.println("拼接后得到的字符串:" + str); + } + + in.close(); + proc.waitFor(); + + + str = str.substring(0, str.length() - 1); + + //去除最后一个逗号 + str = str.substring(str.lastIndexOf(",")); + //截取第一个逗号后的字符串 + str = str.split(",")[1]; + + return str; + } catch (Exception e) { + e.printStackTrace(); + } + return null; + } + + + /** + * 动态传参调用Python脚本:历史模拟 + * + * @param pyPath + * @param path + * @param params + * @throws IOException + */ + public static String getCodeByHis(String pyPath, String path, String... params) throws IOException { + try { + + String[] args = new String[2 + params.length]; + args[0] = pyPath; + args[1] = path; + for (int x = 0; x < args.length; x++) { + if (x > 1) { + args[x] = params[x - 2]; + } + } + //执行py文件 + Process proc = Runtime.getRuntime().exec(args); + BufferedReader in = new BufferedReader(new InputStreamReader(proc.getInputStream())); + String line = null; + + String str = ""; + + + while ((line = in.readLine()) != null) { + System.out.println(line); + str += line; + } + + in.close(); + proc.waitFor(); + + return str; + + + } catch (Exception e) { + e.printStackTrace(); + } + return null; + } + + /** + * test + * + * @param args + * @throws Exception + */ + public static void main(String[] args) throws Exception { + /** + * photo 次数 天数 列表 + * getVarByHistory 置信水平(小数) 天数 列表 + * getVar 置信水平(小数) 天数 列表 + */ + //D:\myTool\py\python.exe + //I为次数,t是天数,Z是传入进来的下单数*期货合约面值,arf为置信水平,ziliao为传入列表 + String print = getCodeByHis(path, historyPath, + "0.95", "90", "6.3792,6.3756"); + + System.out.println(">>>>>>>最后结果:" + print); + } + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/common/utils/trading_rules/TradingDay.java b/src/main/java/com/msdw/tms/common/utils/trading_rules/TradingDay.java new file mode 100644 index 0000000..17c70bc --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/trading_rules/TradingDay.java @@ -0,0 +1,167 @@ +package com.msdw.tms.common.utils.trading_rules; + +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Calendar; +import java.util.List; + +/** + * @ProjectName: code_generator + * @Package: com.create.generator + * @ClassName: test1 + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/13 14:08 + * @UpdateDate: 2020/10/13 14:08 + * @Version: 1.0 + */ +public class TradingDay { + /* + * 计算某年某月第几个星期几的日期 + * param wek 星期几 + * param num 第几个,10为最后一个 + */ + public static String getTheDate(int year, int month, int wek, int num) { + Calendar time = Calendar.getInstance(); + time.set(Calendar.YEAR, year); + time.set(Calendar.MONTH, month - 1);// 注意,Calendar对象默认一月为0 + int day = time.getActualMaximum(Calendar.DAY_OF_MONTH);// 本月份的天数 + List list = new ArrayList<>(); + String dateTime = ""; + for (int i = 1; i <= day; i++) { + time.set(Calendar.DAY_OF_MONTH, i); + int weekd = time.get(Calendar.DAY_OF_WEEK) - 1;// 注意,Calendar对象默认星期天为1 + if (wek >= 1 && wek <= 7) { + if (wek < 7) { + if (weekd == wek) { + String aaa = year + "-" + month + "-" + i; + list.add(aaa); + } + } else { + if (weekd == 0) { + String aaa = year + "-" + month + "-" + i; + list.add(aaa); + } + } + + } + + } + if (list.size() > 0 && num <= 5) { + dateTime = list.get(num - 1); + } else if (num == 10) { + dateTime = list.get(list.size() - 1); + } + return dateTime; + } + + + /** + * 查询某年某月最后一天星期几 + * + * @param yearMonth + * @return + */ + public static String getLastDayOfMonth(String yearMonth) { + int year = Integer.parseInt(yearMonth.split("-")[0]); //年 + int month = Integer.parseInt(yearMonth.split("-")[1]); //月 + Calendar cal = Calendar.getInstance(); + // 设置年份 + cal.set(Calendar.YEAR, year); + // 设置月份 + // cal.set(Calendar.MONTH, month - 1); + cal.set(Calendar.MONTH, month); //设置当前月的上一个月 + // 获取某月最大天数 + //int lastDay = cal.getActualMaximum(Calendar.DATE); + int lastDay = cal.getMinimum(Calendar.DATE); //获取月份中的最小值,即第一天 + // 设置日历中月份的最大天数 + //cal.set(Calendar.DAY_OF_MONTH, lastDay); + cal.set(Calendar.DAY_OF_MONTH, lastDay - 1); //上月的第一天减去1就是当月的最后一天 + // 格式化日期 + SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); + return sdf.format(cal.getTime()); + } + + + /** + * 判断当前日期是星期几
+ *
+ * + * @param pTime 修要判断的时间
+ * @return dayForWeek 判断结果
+ * @Exception 发生异常
+ */ + public static int dayForWeek(String pTime) throws Exception { + SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd"); + Calendar c = Calendar.getInstance(); + c.setTime(format.parse(pTime)); + int dayForWeek = 0; + if (c.get(Calendar.DAY_OF_WEEK) == 1) { + dayForWeek = 7; + } else { + dayForWeek = c.get(Calendar.DAY_OF_WEEK) - 1; + } + return dayForWeek; + } + + + public static String getLastWeek(String time) { + //查询本月最后一天星期几 + try { + + //last_day:查询本月最后一天是多少号 2020-10 + String last_day = getLastDayOfMonth(time); + + + //截取年 + String year = last_day.substring(0, 4); + //截取月 + String month = last_day.substring(5, 7); + //截取日 + String day = last_day.substring(8); + + + //查询最后一天是周几 + Integer weekday = dayForWeek(getLastDayOfMonth(time)); + + + switch (weekday) { + case 1: + weekday = Integer.valueOf(day) - 3; + break; + case 2: + weekday = Integer.valueOf(day) - 4; + break; + case 3: + weekday = Integer.valueOf(day) - 5; + break; + case 4: + weekday = Integer.valueOf(day) - 6; + break; + case 5: + //等于周五不变 + break; + case 6: + weekday = Integer.valueOf(day) - 1; + break; + case 7: + weekday = Integer.valueOf(day) - 2; + break; + } + + return year + "-" + month + "-" + String.valueOf(weekday); + + } catch (Exception e) { + e.printStackTrace(); + return null; + } + } + + public static void main(String[] args) { + //2020年10月的第三个星期日 + //2020年的第3个星期 3前两个营业日 + //System.out.println(getTheDate(2020, 9, 3, 3)); + + System.out.println("本月最后一周星期五:" + getLastWeek("2020-10")); + } +} diff --git a/src/main/java/com/msdw/tms/common/utils/trading_rules/algorithm.java b/src/main/java/com/msdw/tms/common/utils/trading_rules/algorithm.java new file mode 100644 index 0000000..be52ba0 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/utils/trading_rules/algorithm.java @@ -0,0 +1,434 @@ +package com.msdw.tms.common.utils.trading_rules; + +import java.math.BigDecimal; +import java.text.ParseException; +import java.text.SimpleDateFormat; +import java.util.Calendar; +import java.util.Date; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.common.utils.chuanda + * @ClassName: algorithm + * @Description: 川大交易规则算法 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/12 10:04 + * @UpdateDate: 2020/10/12 10:04 + * @Version: 1.0 + */ +public class algorithm { + + /** + *可交易数量: + *1、多方情况下,可交易数量为多方合约持仓总量 + *2、空方情况下,可交易数量为空方合约持仓总量 + */ + + //--------------------配置期货头寸START------------------------ + + /** + * 预冻结保证金: 预计冻结保证金 = 外汇期货合约价格 * 合约面值 * 保证金比例 + * 合约价格: contractPrice + * 合约面值: contractFaceValue + * 保证金比例: bond + */ + public static BigDecimal getPreFreezingMargin(BigDecimal contractPrice, BigDecimal contractFaceValue, String name) { + + try { + //保证金比例 + BigDecimal bond = new BigDecimal(0); + + + switch (name.substring(0, 8)) { + case algorithm.AUDAndRMB: + bond = algorithm.AUDAndRMBBond; + break; + case algorithm.EURAndRMB: + bond = algorithm.EURAndRMBBond; + break; + case algorithm.JPYAndRMB: + bond = algorithm.JPYAndRMBBond; + break; + case algorithm.RMBAndUSD: + bond = algorithm.RMBAndUSDBond; + break; + case algorithm.USDAndRMB: + bond = algorithm.USDAndRMBBond; + break; + + } + BigDecimal amount = contractPrice.multiply(contractFaceValue).multiply(bond); + return amount; + } catch (Exception e) { + e.printStackTrace(); + return new BigDecimal(-1); + } + } + + + /** + * 保证金: 保证金 = 外汇期货合约价格 * 合约面值 * 委托数量 * 保证金比例 + * 合约价格: contractPrice + * 合约面值: contractFaceValue + * 保证金比例: bond + * 手数: handCout + */ + public static BigDecimal freezingMargin(BigDecimal contractPrice, BigDecimal contractFaceValue, BigDecimal handCout, String name) { + + try { + //保证金比例 + BigDecimal bond = new BigDecimal(0.00); + + + switch (name.substring(0, 8)) { + case algorithm.AUDAndRMB: + bond = algorithm.AUDAndRMBBond; + break; + case algorithm.EURAndRMB: + bond = algorithm.EURAndRMBBond; + break; + case algorithm.JPYAndRMB: + bond = algorithm.JPYAndRMBBond; + break; + case algorithm.RMBAndUSD: + bond = algorithm.RMBAndUSDBond; + break; + case algorithm.USDAndRMB: + bond = algorithm.USDAndRMBBond; + break; + + } + BigDecimal amount = contractPrice.multiply(contractFaceValue).multiply(handCout).multiply(bond); + return amount; + } catch (Exception e) { + e.printStackTrace(); + return new BigDecimal(-1); + } + } + + /** + * 预冻结手续费: 预冻结手续费 = 委托数量(手)* 费用标准 + * 委托数量: commissionQuantity + * 费用标准: cost + */ + public static BigDecimal getPreFreezingServiceCharge(BigDecimal commissionQuantity) { + //暂定0.5% + BigDecimal cost = new BigDecimal(0.005); + BigDecimal amount = commissionQuantity.multiply(cost); + return amount; + } + + /** + * 冻结资金:冻结资金=预计冻结保证金+预冻结手续费 + * 预计冻结保证金: preFreezingFunds + * 预冻结手续费: preFreezingCharges + */ + public static double getFreezingFunds(double preFreezingFunds, double preFreezingCharges) { + return preFreezingFunds + preFreezingCharges; + } + + /** + * 需要资金:需要资金 = 委托价格 * 每手交易单位 * 委托数量(手数) + * 委托价格: commissionPrice + * 每手交易单位 : tradingUnit + * 委托数量: commissionQuantity + */ + public static BigDecimal getFundsRequired(BigDecimal commissionPrice, BigDecimal commissionQuantity, Integer tradingUnit) { + return commissionPrice.multiply(commissionQuantity).multiply(BigDecimal.valueOf(tradingUnit)); + } + + //--------------------配置期货头寸 END------------------------ + + //--------外汇期货每手价格 交易单位 START--------- + /** + * 欧元兑人民币期货(EUR) + */ + public static final int EURExchangeRMB = 50000;//欧元 + + /** + * 日元兑人民币期货(JPY) + */ + public static final int JPYExchangeRMB = 6000000;//6000000 日元 + + /** + * 澳元兑人民币期货(AUD) + */ + public static final int AUDExchangeRMB = 80000;//80000 澳元 + + /** + * 人民币兑美元期货(RMB) + */ + public static final int RMBExchangeUSD = 30000;//30000 人民币 + /** + * 美元兑人民币期货(USD) + */ + public static final int USDExchangeRMB = 100000;//100000 美元 + + //--------外汇期货每手价格 END--------- + + + //期货类型 + + /** + * 欧元兑人民币期货(EUR) + */ + public static final String EURAndRMB = "欧元兑人民币期货"; + + /** + * 日元兑人民币期货(JPY) + */ + public static final String JPYAndRMB = "日元兑人民币期货"; + + /** + * 澳元兑人民币期货(AUD) + */ + public static final String AUDAndRMB = "澳元兑人民币期货"; + + /** + * 人民币兑美元期货(RMB) + */ + + public static final String RMBAndUSD = "人民币兑美元期货"; + /** + * 美元兑人民币期货(USD) + */ + public static final String USDAndRMB = "美元兑人民币期货"; + + + /** + * 保证金设置 + */ + + /** + * 欧元兑人民币期货(EUR) + */ + public static final BigDecimal EURAndRMBBond = new BigDecimal(0.023); + /** + * 日元兑人民币期货(JPY) + */ + public static final BigDecimal JPYAndRMBBond = new BigDecimal(0.035); + /** + * 澳元兑人民币期货(AUD) + */ + public static final BigDecimal AUDAndRMBBond = new BigDecimal(0.031); + /** + * 人民币兑美元期货(RMB) + */ + public static final BigDecimal RMBAndUSDBond = new BigDecimal(0.17); + /** + * 美元兑人民币期货(USD) + */ + public static final BigDecimal USDAndRMBBond = new BigDecimal(0.018); + + + /** + * 浮动盈亏=(当天结算价—昨日结算价)* 委托数量 * 合约面值(1份合约=一手,一手=?吨) + *

+ * 当天结算价: todayPrice + * 昨日结算价 : yesterdayPrice + * 合约面值: contractFaceValue + * 委托数量: commissionQuantity + */ + public static BigDecimal getFloatingPL(BigDecimal todayPrice, BigDecimal yesterdayPrice, BigDecimal commissionQuantity, Integer contractFaceValue) { + return todayPrice.subtract(yesterdayPrice).multiply(commissionQuantity).multiply(BigDecimal.valueOf(contractFaceValue)); + } + + /*public static void main(String[] args) { + BigDecimal pl = getFloatingPL(new BigDecimal(6.826), new BigDecimal(6.84), new BigDecimal(1), 100000); + System.out.println(pl); + }*/ + + /** + * 盈亏=(平仓时的外汇汇率价格-下单时外汇汇率价格) * 委托数量 * 合约面值。 + *

+ * commissionQuantity:委托数量 + * contractFaceValue: 合约面值 + * + * @return + */ + /*public static BigDecimal getPL(BigDecimal price1, BigDecimal price2, BigDecimal commissionQuantity, BigDecimal contractFaceValue) { + return price1.subtract(price2).multiply(commissionQuantity).multiply(contractFaceValue).setScale(2, BigDecimal.ROUND_DOWN); + }*/ + + /** + * 外汇期货盈亏=(平仓时的外汇汇率价格-下单时外汇汇率价格) * 委托数量 * 合约面值。 + *

+ * commissionQuantity:委托数量 + * contractFaceValue: 合约面值 + * + * @return + */ + public static BigDecimal get_wh_pl(BigDecimal price1, BigDecimal price2, BigDecimal commissionQuantity, BigDecimal contractFaceValue) { + // return price1.subtract(price2).multiply(commissionQuantity).multiply(contractFaceValue).setScale(2, BigDecimal.ROUND_DOWN); + return price1.subtract(price2).multiply(commissionQuantity).multiply(contractFaceValue); + } + + + /** + * 资产市值 = (开仓或者平仓) * 委托数量 * 合约面值(1份合约=一手,一手=?吨) + * commissionQuantity:委托数量 + * contractFaceValue: 合约面值 + * 市值 = 外汇期货合约价格 * 委托数量 * 合约面值 + */ + public static BigDecimal getMarketValue(BigDecimal price, Integer commissionQuantity, Integer contractFaceValue) { + return price.multiply(BigDecimal.valueOf(commissionQuantity)).multiply(BigDecimal.valueOf(contractFaceValue)); + } + + + /** + * 调仓操作 + * 加仓市值 = 外汇期货合约价格*(原持仓量 + 本次调仓量)* 合约面值, + * price:外汇期货合约价格 + * old:原持仓量 + * news:本次调仓量 + * contractFaceValue: 合约面值 + */ + public static BigDecimal getMarketValueByAddition(BigDecimal price, BigDecimal old, BigDecimal news, Integer contractFaceValue) { + BigDecimal a = price.multiply(new BigDecimal(contractFaceValue)); + BigDecimal b = old.add(news); + return a.multiply(b); + } + + + /** + * 调仓操作 + * 减仓市值 = 外汇期货合约价格*(原持仓量 - 本次调仓量)* 合约面值, + * price:外汇期货合约价格 + * old:原持仓量 + * news:本次调仓量 + * contractFaceValue: 合约面值 + */ + public static BigDecimal getMarketValueBySubtraction(BigDecimal price, BigDecimal old, BigDecimal news, Integer contractFaceValue) { + BigDecimal a = price.multiply(new BigDecimal(contractFaceValue)); + BigDecimal b = old.subtract(news); + return a.multiply(b); + } + + + /** + * 外币 = 汇率 * 合约面值 + * + * @param price 汇率 + * @param contractFaceValue 合约面值 + * @return + */ + public static BigDecimal getForeignCurrency(BigDecimal price, BigDecimal contractFaceValue) { + return price.multiply(contractFaceValue); + } + +//----------------最后交易日------------------ + + /** + * 获取最后交易日 + * (1~4) + * * 1.欧元兑人民币期货 + * * 2.日元兑人民币期货 + * * 3.澳元兑人民币期货 + * * 4.人民币兑美元期货 + * *

+ * * 5.美元兑人民币期货 最后结算日之前两个营业日 + * + * @param name + * @return 获取最后交易日 + */ + + public static String getLastTradingDay(String name) { + + + try { + String getName = name.substring(0, 8); + //年份 + String year = name.substring(8, 10); + //月份 + String month = name.substring(10); + + //最后交易日为:合约月份的第3个星期3前两个营业日 + if (getName.equals(algorithm.AUDAndRMB) || getName.equals(algorithm.EURAndRMB) || getName.equals(algorithm.JPYAndRMB) || getName.equals(algorithm.RMBAndUSD)) { + + + //获取某年某月的第3个星期天的星期三 + String reTime = TradingDay.getTheDate(Integer.valueOf("20" + year), Integer.valueOf(month), 3, 3); + + SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); + + Date date = sdf.parse(reTime); + + String lastTime = getAddDate(date); + + return lastTime; + } else if (getName.equals(algorithm.USDAndRMB)) { + + + String times = "20" + year + "-" + month; + //最后结算日之前两个营业日(本月的最后一个周五) + String lastWeek = TradingDay.getLastWeek(times); + return lastWeek; + + } + } catch (ParseException e) { + e.printStackTrace(); + } + return null; + + } + + + //得到添加n天后的时间字符串 + public static String getAddDate(Date date) { + try { + SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd"); + + Calendar calendar = Calendar.getInstance(); + calendar.setTime(date); + + //减2天 + calendar.setTime(date); + calendar.add(Calendar.DAY_OF_MONTH, -2); + String T2 = format.format(calendar.getTime()); + return T2;//返回时间字符串 + + } catch (Exception e) { + e.printStackTrace(); + return null; + } + } + + /** + * 根据获取的合约名称设置交易保证金 + * + * @param name + * @return + */ + public static BigDecimal getBond(String name) { + + try { + //保证金比例 + BigDecimal bond = new BigDecimal(0); + + switch (name.substring(0, 8)) { + case algorithm.AUDAndRMB: + bond = algorithm.AUDAndRMBBond; + break; + case algorithm.EURAndRMB: + bond = algorithm.EURAndRMBBond; + break; + case algorithm.JPYAndRMB: + bond = algorithm.JPYAndRMBBond; + break; + case algorithm.RMBAndUSD: + bond = algorithm.RMBAndUSDBond; + break; + case algorithm.USDAndRMB: + bond = algorithm.USDAndRMBBond; + break; + } + return bond; + } catch (Exception e) { + e.printStackTrace(); + return new BigDecimal(-1); + } + } + + +} + diff --git a/src/main/java/com/msdw/tms/common/xss/HTMLFilter.java b/src/main/java/com/msdw/tms/common/xss/HTMLFilter.java new file mode 100644 index 0000000..6f1df38 --- /dev/null +++ b/src/main/java/com/msdw/tms/common/xss/HTMLFilter.java @@ -0,0 +1,530 @@ +package com.msdw.tms.common.xss; + +import java.util.*; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.ConcurrentMap; +import java.util.logging.Logger; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +/** + * + * HTML filtering utility for protecting against XSS (Cross Site Scripting). + * + * This code is licensed LGPLv3 + * + * This code is a Java port of the original work in PHP by Cal Hendersen. + * http://code.iamcal.com/php/lib_filter/ + * + * The trickiest part of the translation was handling the differences in regex handling + * between PHP and Java. These resources were helpful in the process: + * + * http://java.sun.com/j2se/1.4.2/docs/api/java/util/regex/Pattern.html + * http://us2.php.net/manual/en/reference.pcre.pattern.modifiers.php + * http://www.regular-expressions.info/modifiers.html + * + * A note on naming conventions: instance variables are prefixed with a "v"; global + * constants are in all caps. + * + * Sample use: + * String input = ... + * String clean = new HTMLFilter().filter( input ); + * + * The class is not thread safe. Create a new instance if in doubt. + * + * If you find bugs or have suggestions on improvement (especially regarding + * performance), please contact us. The latest version of this + * source, and our contact details, can be found at http://xss-html-filter.sf.net + * + * @author Joseph O'Connell + * @author Cal Hendersen + * @author Michael Semb Wever + */ +public final class HTMLFilter { + + /** regex flag union representing /si modifiers in php **/ + private static final int REGEX_FLAGS_SI = Pattern.CASE_INSENSITIVE | Pattern.DOTALL; + private static final Pattern P_COMMENTS = Pattern.compile("", Pattern.DOTALL); + private static final Pattern P_COMMENT = Pattern.compile("^!--(.*)--$", REGEX_FLAGS_SI); + private static final Pattern P_TAGS = Pattern.compile("<(.*?)>", Pattern.DOTALL); + private static final Pattern P_END_TAG = Pattern.compile("^/([a-z0-9]+)", REGEX_FLAGS_SI); + private static final Pattern P_START_TAG = Pattern.compile("^([a-z0-9]+)(.*?)(/?)$", REGEX_FLAGS_SI); + private static final Pattern P_QUOTED_ATTRIBUTES = Pattern.compile("([a-z0-9]+)=([\"'])(.*?)\\2", REGEX_FLAGS_SI); + private static final Pattern P_UNQUOTED_ATTRIBUTES = Pattern.compile("([a-z0-9]+)(=)([^\"\\s']+)", REGEX_FLAGS_SI); + private static final Pattern P_PROTOCOL = Pattern.compile("^([^:]+):", REGEX_FLAGS_SI); + private static final Pattern P_ENTITY = Pattern.compile("&#(\\d+);?"); + private static final Pattern P_ENTITY_UNICODE = Pattern.compile("&#x([0-9a-f]+);?"); + private static final Pattern P_ENCODE = Pattern.compile("%([0-9a-f]{2});?"); + private static final Pattern P_VALID_ENTITIES = Pattern.compile("&([^&;]*)(?=(;|&|$))"); + private static final Pattern P_VALID_QUOTES = Pattern.compile("(>|^)([^<]+?)(<|$)", Pattern.DOTALL); + private static final Pattern P_END_ARROW = Pattern.compile("^>"); + private static final Pattern P_BODY_TO_END = Pattern.compile("<([^>]*?)(?=<|$)"); + private static final Pattern P_XML_CONTENT = Pattern.compile("(^|>)([^<]*?)(?=>)"); + private static final Pattern P_STRAY_LEFT_ARROW = Pattern.compile("<([^>]*?)(?=<|$)"); + private static final Pattern P_STRAY_RIGHT_ARROW = Pattern.compile("(^|>)([^<]*?)(?=>)"); + private static final Pattern P_AMP = Pattern.compile("&"); + private static final Pattern P_QUOTE = Pattern.compile("<"); + private static final Pattern P_LEFT_ARROW = Pattern.compile("<"); + private static final Pattern P_RIGHT_ARROW = Pattern.compile(">"); + private static final Pattern P_BOTH_ARROWS = Pattern.compile("<>"); + + // @xxx could grow large... maybe use sesat's ReferenceMap + private static final ConcurrentMap P_REMOVE_PAIR_BLANKS = new ConcurrentHashMap(); + private static final ConcurrentMap P_REMOVE_SELF_BLANKS = new ConcurrentHashMap(); + + /** set of allowed html elements, along with allowed attributes for each element **/ + private final Map> vAllowed; + /** counts of open tags for each (allowable) html element **/ + private final Map vTagCounts = new HashMap(); + + /** html elements which must always be self-closing (e.g. "") **/ + private final String[] vSelfClosingTags; + /** html elements which must always have separate opening and closing tags (e.g. "") **/ + private final String[] vNeedClosingTags; + /** set of disallowed html elements **/ + private final String[] vDisallowed; + /** attributes which should be checked for valid protocols **/ + private final String[] vProtocolAtts; + /** allowed protocols **/ + private final String[] vAllowedProtocols; + /** tags which should be removed if they contain no content (e.g. "" or "") **/ + private final String[] vRemoveBlanks; + /** entities allowed within html markup **/ + private final String[] vAllowedEntities; + /** flag determining whether comments are allowed in input String. */ + private final boolean stripComment; + private final boolean encodeQuotes; + private boolean vDebug = false; + /** + * flag determining whether to try to make tags when presented with "unbalanced" + * angle brackets (e.g. "" becomes " text "). If set to false, + * unbalanced angle brackets will be html escaped. + */ + private final boolean alwaysMakeTags; + + /** Default constructor. + * + */ + public HTMLFilter() { + vAllowed = new HashMap<>(); + + final ArrayList a_atts = new ArrayList(); + a_atts.add("href"); + a_atts.add("target"); + vAllowed.put("a", a_atts); + + final ArrayList img_atts = new ArrayList(); + img_atts.add("src"); + img_atts.add("width"); + img_atts.add("height"); + img_atts.add("alt"); + vAllowed.put("img", img_atts); + + final ArrayList no_atts = new ArrayList(); + vAllowed.put("b", no_atts); + vAllowed.put("strong", no_atts); + vAllowed.put("i", no_atts); + vAllowed.put("em", no_atts); + + vSelfClosingTags = new String[]{"img"}; + vNeedClosingTags = new String[]{"a", "b", "strong", "i", "em"}; + vDisallowed = new String[]{}; + vAllowedProtocols = new String[]{"http", "mailto", "https"}; // no ftp. + vProtocolAtts = new String[]{"src", "href"}; + vRemoveBlanks = new String[]{"a", "b", "strong", "i", "em"}; + vAllowedEntities = new String[]{"amp", "gt", "lt", "quot"}; + stripComment = true; + encodeQuotes = true; + alwaysMakeTags = true; + } + + /** Set debug flag to true. Otherwise use default settings. See the default constructor. + * + * @param debug turn debug on with a true argument + */ + public HTMLFilter(final boolean debug) { + this(); + vDebug = debug; + + } + + /** Map-parameter configurable constructor. + * + * @param conf map containing configuration. keys match field names. + */ + public HTMLFilter(final Map conf) { + + assert conf.containsKey("vAllowed") : "configuration requires vAllowed"; + assert conf.containsKey("vSelfClosingTags") : "configuration requires vSelfClosingTags"; + assert conf.containsKey("vNeedClosingTags") : "configuration requires vNeedClosingTags"; + assert conf.containsKey("vDisallowed") : "configuration requires vDisallowed"; + assert conf.containsKey("vAllowedProtocols") : "configuration requires vAllowedProtocols"; + assert conf.containsKey("vProtocolAtts") : "configuration requires vProtocolAtts"; + assert conf.containsKey("vRemoveBlanks") : "configuration requires vRemoveBlanks"; + assert conf.containsKey("vAllowedEntities") : "configuration requires vAllowedEntities"; + + vAllowed = Collections.unmodifiableMap((HashMap>) conf.get("vAllowed")); + vSelfClosingTags = (String[]) conf.get("vSelfClosingTags"); + vNeedClosingTags = (String[]) conf.get("vNeedClosingTags"); + vDisallowed = (String[]) conf.get("vDisallowed"); + vAllowedProtocols = (String[]) conf.get("vAllowedProtocols"); + vProtocolAtts = (String[]) conf.get("vProtocolAtts"); + vRemoveBlanks = (String[]) conf.get("vRemoveBlanks"); + vAllowedEntities = (String[]) conf.get("vAllowedEntities"); + stripComment = conf.containsKey("stripComment") ? (Boolean) conf.get("stripComment") : true; + encodeQuotes = conf.containsKey("encodeQuotes") ? (Boolean) conf.get("encodeQuotes") : true; + alwaysMakeTags = conf.containsKey("alwaysMakeTags") ? (Boolean) conf.get("alwaysMakeTags") : true; + } + + private void reset() { + vTagCounts.clear(); + } + + private void debug(final String msg) { + if (vDebug) { + Logger.getAnonymousLogger().info(msg); + } + } + + //--------------------------------------------------------------- + // my versions of some PHP library functions + public static String chr(final int decimal) { + return String.valueOf((char) decimal); + } + + public static String htmlSpecialChars(final String s) { + String result = s; + result = regexReplace(P_AMP, "&", result); + result = regexReplace(P_QUOTE, """, result); + result = regexReplace(P_LEFT_ARROW, "<", result); + result = regexReplace(P_RIGHT_ARROW, ">", result); + return result; + } + + //--------------------------------------------------------------- + /** + * given a user submitted input String, filter out any invalid or restricted + * html. + * + * @param input text (i.e. submitted by a user) than may contain html + * @return "clean" version of input, with only valid, whitelisted html elements allowed + */ + public String filter(final String input) { + reset(); + String s = input; + + debug("************************************************"); + debug(" INPUT: " + input); + + s = escapeComments(s); + debug(" escapeComments: " + s); + + s = balanceHTML(s); + debug(" balanceHTML: " + s); + + s = checkTags(s); + debug(" checkTags: " + s); + + s = processRemoveBlanks(s); + debug("processRemoveBlanks: " + s); + + s = validateEntities(s); + debug(" validateEntites: " + s); + + debug("************************************************\n\n"); + return s; + } + + public boolean isAlwaysMakeTags(){ + return alwaysMakeTags; + } + + public boolean isStripComments(){ + return stripComment; + } + + private String escapeComments(final String s) { + final Matcher m = P_COMMENTS.matcher(s); + final StringBuffer buf = new StringBuffer(); + if (m.find()) { + final String match = m.group(1); //(.*?) + m.appendReplacement(buf, Matcher.quoteReplacement("")); + } + m.appendTail(buf); + + return buf.toString(); + } + + private String balanceHTML(String s) { + if (alwaysMakeTags) { + // + // try and form html + // + s = regexReplace(P_END_ARROW, "", s); + s = regexReplace(P_BODY_TO_END, "<$1>", s); + s = regexReplace(P_XML_CONTENT, "$1<$2", s); + + } else { + // + // escape stray brackets + // + s = regexReplace(P_STRAY_LEFT_ARROW, "<$1", s); + s = regexReplace(P_STRAY_RIGHT_ARROW, "$1$2><", s); + + // + // the last regexp causes '<>' entities to appear + // (we need to do a lookahead assertion so that the last bracket can + // be used in the next pass of the regexp) + // + s = regexReplace(P_BOTH_ARROWS, "", s); + } + + return s; + } + + private String checkTags(String s) { + Matcher m = P_TAGS.matcher(s); + + final StringBuffer buf = new StringBuffer(); + while (m.find()) { + String replaceStr = m.group(1); + replaceStr = processTag(replaceStr); + m.appendReplacement(buf, Matcher.quoteReplacement(replaceStr)); + } + m.appendTail(buf); + + s = buf.toString(); + + // these get tallied in processTag + // (remember to reset before subsequent calls to filter method) + for (String key : vTagCounts.keySet()) { + for (int ii = 0; ii < vTagCounts.get(key); ii++) { + s += ""; + } + } + + return s; + } + + private String processRemoveBlanks(final String s) { + String result = s; + for (String tag : vRemoveBlanks) { + if(!P_REMOVE_PAIR_BLANKS.containsKey(tag)){ + P_REMOVE_PAIR_BLANKS.putIfAbsent(tag, Pattern.compile("<" + tag + "(\\s[^>]*)?>")); + } + result = regexReplace(P_REMOVE_PAIR_BLANKS.get(tag), "", result); + if(!P_REMOVE_SELF_BLANKS.containsKey(tag)){ + P_REMOVE_SELF_BLANKS.putIfAbsent(tag, Pattern.compile("<" + tag + "(\\s[^>]*)?/>")); + } + result = regexReplace(P_REMOVE_SELF_BLANKS.get(tag), "", result); + } + + return result; + } + + private static String regexReplace(final Pattern regex_pattern, final String replacement, final String s) { + Matcher m = regex_pattern.matcher(s); + return m.replaceAll(replacement); + } + + private String processTag(final String s) { + // ending tags + Matcher m = P_END_TAG.matcher(s); + if (m.find()) { + final String name = m.group(1).toLowerCase(); + if (allowed(name)) { + if (!inArray(name, vSelfClosingTags)) { + if (vTagCounts.containsKey(name)) { + vTagCounts.put(name, vTagCounts.get(name) - 1); + return ""; + } + } + } + } + + // starting tags + m = P_START_TAG.matcher(s); + if (m.find()) { + final String name = m.group(1).toLowerCase(); + final String body = m.group(2); + String ending = m.group(3); + + //debug( "in a starting tag, name='" + name + "'; body='" + body + "'; ending='" + ending + "'" ); + if (allowed(name)) { + String params = ""; + + final Matcher m2 = P_QUOTED_ATTRIBUTES.matcher(body); + final Matcher m3 = P_UNQUOTED_ATTRIBUTES.matcher(body); + final List paramNames = new ArrayList(); + final List paramValues = new ArrayList(); + while (m2.find()) { + paramNames.add(m2.group(1)); //([a-z0-9]+) + paramValues.add(m2.group(3)); //(.*?) + } + while (m3.find()) { + paramNames.add(m3.group(1)); //([a-z0-9]+) + paramValues.add(m3.group(3)); //([^\"\\s']+) + } + + String paramName, paramValue; + for (int ii = 0; ii < paramNames.size(); ii++) { + paramName = paramNames.get(ii).toLowerCase(); + paramValue = paramValues.get(ii); + +// debug( "paramName='" + paramName + "'" ); +// debug( "paramValue='" + paramValue + "'" ); +// debug( "allowed? " + vAllowed.get( name ).contains( paramName ) ); + + if (allowedAttribute(name, paramName)) { + if (inArray(paramName, vProtocolAtts)) { + paramValue = processParamProtocol(paramValue); + } + params += " " + paramName + "=\"" + paramValue + "\""; + } + } + + if (inArray(name, vSelfClosingTags)) { + ending = " /"; + } + + if (inArray(name, vNeedClosingTags)) { + ending = ""; + } + + if (ending == null || ending.length() < 1) { + if (vTagCounts.containsKey(name)) { + vTagCounts.put(name, vTagCounts.get(name) + 1); + } else { + vTagCounts.put(name, 1); + } + } else { + ending = " /"; + } + return "<" + name + params + ending + ">"; + } else { + return ""; + } + } + + // comments + m = P_COMMENT.matcher(s); + if (!stripComment && m.find()) { + return "<" + m.group() + ">"; + } + + return ""; + } + + private String processParamProtocol(String s) { + s = decodeEntities(s); + final Matcher m = P_PROTOCOL.matcher(s); + if (m.find()) { + final String protocol = m.group(1); + if (!inArray(protocol, vAllowedProtocols)) { + // bad protocol, turn into local anchor link instead + s = "#" + s.substring(protocol.length() + 1, s.length()); + if (s.startsWith("#//")) { + s = "#" + s.substring(3, s.length()); + } + } + } + + return s; + } + + private String decodeEntities(String s) { + StringBuffer buf = new StringBuffer(); + + Matcher m = P_ENTITY.matcher(s); + while (m.find()) { + final String match = m.group(1); + final int decimal = Integer.decode(match).intValue(); + m.appendReplacement(buf, Matcher.quoteReplacement(chr(decimal))); + } + m.appendTail(buf); + s = buf.toString(); + + buf = new StringBuffer(); + m = P_ENTITY_UNICODE.matcher(s); + while (m.find()) { + final String match = m.group(1); + final int decimal = Integer.valueOf(match, 16).intValue(); + m.appendReplacement(buf, Matcher.quoteReplacement(chr(decimal))); + } + m.appendTail(buf); + s = buf.toString(); + + buf = new StringBuffer(); + m = P_ENCODE.matcher(s); + while (m.find()) { + final String match = m.group(1); + final int decimal = Integer.valueOf(match, 16).intValue(); + m.appendReplacement(buf, Matcher.quoteReplacement(chr(decimal))); + } + m.appendTail(buf); + s = buf.toString(); + + s = validateEntities(s); + return s; + } + + private String validateEntities(final String s) { + StringBuffer buf = new StringBuffer(); + + // validate entities throughout the string + Matcher m = P_VALID_ENTITIES.matcher(s); + while (m.find()) { + final String one = m.group(1); //([^&;]*) + final String two = m.group(2); //(?=(;|&|$)) + m.appendReplacement(buf, Matcher.quoteReplacement(checkEntity(one, two))); + } + m.appendTail(buf); + + return encodeQuotes(buf.toString()); + } + + private String encodeQuotes(final String s){ + if(encodeQuotes){ + StringBuffer buf = new StringBuffer(); + Matcher m = P_VALID_QUOTES.matcher(s); + while (m.find()) { + final String one = m.group(1); //(>|^) + final String two = m.group(2); //([^<]+?) + final String three = m.group(3); //(<|$) + m.appendReplacement(buf, Matcher.quoteReplacement(one + regexReplace(P_QUOTE, """, two) + three)); + } + m.appendTail(buf); + return buf.toString(); + }else{ + return s; + } + } + + private String checkEntity(final String preamble, final String term) { + + return ";".equals(term) && isValidEntity(preamble) + ? '&' + preamble + : "&" + preamble; + } + + private boolean isValidEntity(final String entity) { + return inArray(entity, vAllowedEntities); + } + + private static boolean inArray(final String s, final String[] array) { + for (String item : array) { + if (item != null && item.equals(s)) { + return true; + } + } + return false; + } + + private boolean allowed(final String name) { + return (vAllowed.isEmpty() || vAllowed.containsKey(name)) && !inArray(name, vDisallowed); + } + + private boolean allowedAttribute(final String name, final String paramName) { + return allowed(name) && (vAllowed.isEmpty() || vAllowed.get(name).contains(paramName)); + } +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/common/xss/SQLFilter.java b/src/main/java/com/msdw/tms/common/xss/SQLFilter.java new file mode 100644 index 0000000..d8e6e5e --- /dev/null +++ b/src/main/java/com/msdw/tms/common/xss/SQLFilter.java @@ -0,0 +1,49 @@ +/** + * Copyright (c) 2016-2019 人人开源 All rights reserved. + * + * https://www.renren.io + * + * 版权所有,侵权必究! + */ + +package com.msdw.tms.common.xss; + +import org.apache.commons.lang.StringUtils; + +/** + * SQL过滤 + * + * @author Mark sunlightcs@gmail.com + */ +public class SQLFilter { + + /** + * SQL注入过滤 + * @param str 待验证的字符串 + */ + public static String sqlInject(String str){ + if(StringUtils.isBlank(str)){ + return null; + } + //去掉'|"|;|\字符 + str = StringUtils.replace(str, "'", ""); + str = StringUtils.replace(str, "\"", ""); + str = StringUtils.replace(str, ";", ""); + str = StringUtils.replace(str, "\\", ""); + + //转换成小写 + str = str.toLowerCase(); + + //非法字符 + String[] keywords = {"master", "truncate", "insert", "select", "delete", "update", "declare", "alter", "drop"}; + + //判断是否包含非法字符 + /*for(String keyword : keywords){ + if(str.indexOf(keyword) != -1){ + throw new RRException("包含非法字符"); + } + }*/ + + return str; + } +} diff --git a/src/main/java/com/msdw/tms/config/AliyunOssConfig.java b/src/main/java/com/msdw/tms/config/AliyunOssConfig.java new file mode 100644 index 0000000..3bc4d74 --- /dev/null +++ b/src/main/java/com/msdw/tms/config/AliyunOssConfig.java @@ -0,0 +1,59 @@ +package com.msdw.tms.config; + +import com.aliyun.oss.OSSClient; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.PropertySource; +import org.springframework.context.annotation.Scope; +import org.springframework.stereotype.Component; + +@Component +@PropertySource(value = "classpath:aliyun.properties") +@Configuration +public class AliyunOssConfig { + + @Value("${oss.endpoint}") + private String endpoint;// oss外网访问域名 + @Value("${oss.accessKeyId}") + private String accessKeyId;// oss中的密匙keyId + @Value("${oss.secretAccessKey}") + private String secretAccessKey;// oss中的密钥 + @Value("${oss.bucketName}") + private String bucketName;// 仓库名称 + @Value("${oss.sufferUrl}") + private String sufferUrl; + @Value("${user.userAvatars}")//用户头像 + private String userAvatars; + + @Bean + @Scope("prototype") + public OSSClient ossClient() { + return new OSSClient(endpoint, accessKeyId, secretAccessKey); + } + + public String getEndpoint() { + return endpoint; + } + + public String getAccessKeyId() { + return accessKeyId; + } + + public String getSecretAccessKey() { + return secretAccessKey; + } + + public String getBucketName() { + return bucketName; + } + + public String getSufferUrl() { + return sufferUrl; + } + + public String getUserAvatars() { + return userAvatars; + } + +} diff --git a/src/main/java/com/msdw/tms/config/CorsConfig.java b/src/main/java/com/msdw/tms/config/CorsConfig.java new file mode 100644 index 0000000..5250d87 --- /dev/null +++ b/src/main/java/com/msdw/tms/config/CorsConfig.java @@ -0,0 +1,19 @@ +package com.msdw.tms.config; + +import org.springframework.context.annotation.Configuration; +import org.springframework.web.servlet.config.annotation.CorsRegistry; +import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; + +@Configuration +public class CorsConfig implements WebMvcConfigurer { + + @Override + public void addCorsMappings(CorsRegistry registry) { + registry.addMapping("/**") + .allowedOrigins("*") + .allowedMethods("GET", "HEAD", "POST", "PUT", "DELETE", "OPTIONS") + .allowCredentials(true) + .maxAge(3600) + .allowedHeaders("*"); + } +} diff --git a/src/main/java/com/msdw/tms/config/MyThreadConfig.java b/src/main/java/com/msdw/tms/config/MyThreadConfig.java new file mode 100644 index 0000000..b7243eb --- /dev/null +++ b/src/main/java/com/msdw/tms/config/MyThreadConfig.java @@ -0,0 +1,25 @@ +package com.msdw.tms.config; + +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +import java.util.concurrent.Executors; +import java.util.concurrent.LinkedBlockingDeque; +import java.util.concurrent.ThreadPoolExecutor; +import java.util.concurrent.TimeUnit; + +//@EnableConfigurationProperties(ThreadPoolConfigProperties.class) +@Configuration +public class MyThreadConfig { + + @Bean + public ThreadPoolExecutor threadPoolExecutor(ThreadPoolConfigProperties pool) { + return new ThreadPoolExecutor(pool.getCoreSize(), + pool.getMaxSize(), + pool.getKeepAliveTime(), + TimeUnit.SECONDS, + new LinkedBlockingDeque<>(100000), + Executors.defaultThreadFactory(), + new ThreadPoolExecutor.AbortPolicy()); + } +} diff --git a/src/main/java/com/msdw/tms/config/MybatisPlusConfig.java b/src/main/java/com/msdw/tms/config/MybatisPlusConfig.java new file mode 100644 index 0000000..e206ea3 --- /dev/null +++ b/src/main/java/com/msdw/tms/config/MybatisPlusConfig.java @@ -0,0 +1,18 @@ +package com.msdw.tms.config; + +import com.baomidou.mybatisplus.extension.plugins.PaginationInterceptor; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +@Configuration +public class MybatisPlusConfig { + /** + * mybatis-plus分页插件 + */ + @Bean + public PaginationInterceptor paginationInterceptor() { + PaginationInterceptor page = new PaginationInterceptor(); + page.setDialectType("mysql"); + return page; + } +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/config/RedisConfig.java b/src/main/java/com/msdw/tms/config/RedisConfig.java new file mode 100644 index 0000000..9a2e5e7 --- /dev/null +++ b/src/main/java/com/msdw/tms/config/RedisConfig.java @@ -0,0 +1,22 @@ +package com.msdw.tms.config; + +import org.springframework.data.redis.core.RedisTemplate; + +import javax.annotation.Resource; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.config + * @ClassName: RedisConfig + * @Description: 功能描述: redis配置 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/29 17:55 + * @UpdateDate: 2020/10/29 17:55 + * @Version: 1.0 + */ +public class RedisConfig { + + @Resource + public RedisTemplate template; + +} diff --git a/src/main/java/com/msdw/tms/config/SwaggerConfig.java b/src/main/java/com/msdw/tms/config/SwaggerConfig.java new file mode 100644 index 0000000..ff3bc49 --- /dev/null +++ b/src/main/java/com/msdw/tms/config/SwaggerConfig.java @@ -0,0 +1,35 @@ +package com.msdw.tms.config; + +import io.swagger.annotations.ApiOperation; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import springfox.documentation.builders.ApiInfoBuilder; +import springfox.documentation.builders.PathSelectors; +import springfox.documentation.builders.RequestHandlerSelectors; +import springfox.documentation.service.ApiInfo; +import springfox.documentation.spi.DocumentationType; +import springfox.documentation.spring.web.plugins.Docket; +import springfox.documentation.swagger2.annotations.EnableSwagger2; + +@Configuration +@EnableSwagger2 +//@Profile({"dev","test"}) +//@ConditionalOnProperty(name = "swagger.enable", havingValue = "true") +public class SwaggerConfig { + + @Bean + public Docket productApi() { + return new Docket(DocumentationType.SWAGGER_2) + .apiInfo(apiInfo()) + .select()//添加ApiOperiation注解的被扫描 + .apis(RequestHandlerSelectors.withMethodAnnotation(ApiOperation.class)) + .paths(PathSelectors.any()) + .build(); + } + + private ApiInfo apiInfo() { + return new ApiInfoBuilder().title("测评管理").description("测评管理") + .version("1.0").build(); + } + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/config/ThreadPoolConfigProperties.java b/src/main/java/com/msdw/tms/config/ThreadPoolConfigProperties.java new file mode 100644 index 0000000..a704f0a --- /dev/null +++ b/src/main/java/com/msdw/tms/config/ThreadPoolConfigProperties.java @@ -0,0 +1,14 @@ +package com.msdw.tms.config; + +import lombok.Data; +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.stereotype.Component; + +@ConfigurationProperties(prefix = "tms.thread") +@Component +@Data +public class ThreadPoolConfigProperties { + private Integer coreSize; + private Integer maxSize; + private Integer keepAliveTime; +} diff --git a/src/main/java/com/msdw/tms/config/WebConfigurer.java b/src/main/java/com/msdw/tms/config/WebConfigurer.java new file mode 100644 index 0000000..879f263 --- /dev/null +++ b/src/main/java/com/msdw/tms/config/WebConfigurer.java @@ -0,0 +1,37 @@ +package com.msdw.tms.config; + +import com.msdw.tms.interceptor.AuthorizedAspect; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.annotation.Configuration; +import org.springframework.web.servlet.config.annotation.InterceptorRegistry; +import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.config + * @ClassName: WebConfigurer + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/22 11:21 + * @UpdateDate: 2020/10/22 11:21 + * @Version: 1.0 + */ +@Configuration +public class WebConfigurer implements WebMvcConfigurer { + + @Autowired + AuthorizedAspect authorizedAspect; + + /** + * 这个方法用来添加拦截器,我们自己写好的拦截器需要通过这里添加注册才能生效 + * 静态资源不能直接写/static/**,这样还是会被拦截器拦截 + * 排除注册与一些无须登录可看的页面请求 + */ + @Override + public void addInterceptors(InterceptorRegistry registry) { + /*registry.addInterceptor(authorizedAspect).addPathPatterns("/**") + .excludePathPatterns("/swagger-ui.html") + .excludePathPatterns("");*/ + } +} + diff --git a/src/main/java/com/msdw/tms/controller/CdBankStatementsController.java b/src/main/java/com/msdw/tms/controller/CdBankStatementsController.java new file mode 100644 index 0000000..9d55985 --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/CdBankStatementsController.java @@ -0,0 +1,324 @@ +package com.msdw.tms.controller; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.msdw.tms.common.utils.PageUtils; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.TokenUtil; +import com.msdw.tms.common.utils.trading_rules.algorithm; +import com.msdw.tms.entity.CdFuturesConfigureEntity; +import com.msdw.tms.entity.CdUserAssetsEntity; +import com.msdw.tms.entity.CdUserOptionAccountEntity; +import com.msdw.tms.entity.QhscGangjiaosuoEntity; +import com.msdw.tms.model.req.CdGetAccountInfoReq; +import com.msdw.tms.model.req.CdTransferRecordsReq; +import com.msdw.tms.model.req.CdTransferReq; +import com.msdw.tms.model.resp.CdAccountInfoResp; +import com.msdw.tms.model.resp.CdTransferOutResp; +import com.msdw.tms.model.resp.CdTransferRecordsResp; +import com.msdw.tms.service.*; +import com.msdw.tms.service.impl.CdFuturesConfigureServiceImpl; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiOperation; +import io.swagger.annotations.ApiParam; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.redis.core.RedisTemplate; +import org.springframework.web.bind.annotation.*; + +import java.math.BigDecimal; +import java.util.List; + + +/** + * @描述:账户流水记录控制类 + * @作者: Rong + * @日期: 2020-10-15 + */ +@RestController +@RequestMapping("/chuanda/bankStatements") +@Api(value = "API - 川大:账户相关操作", tags = "川大子系统:账户相关操作") +public class CdBankStatementsController { + + @Autowired + public CdBankStatementsService service; + + @Autowired + public CdUserOptionAccountService optionAccountService; + + @Autowired + public CdUserAssetsService assetsService; + + @Autowired + public CdFuturesConfigureService configureService; + + @Autowired + public QhscGangjiaosuoService gangjiaosuoService; + @Autowired + public RedisTemplate template; + + /** + * @描述: 银行/期货期权账户 转账操作 + * @作者: Rong + * @日期: 2020-10-15 + **/ + @PostMapping("/transfer") + @ApiOperation(value = "(银行/期货期权账户)转账操作") + public R save(@RequestBody CdTransferReq req) { + int userId = TokenUtil.getUserId(); + //查询银行账户余额 + CdUserAssetsEntity assets = assetsService.userAssetsByUerId(userId); + //查询期权账户余额 + CdUserOptionAccountEntity account = optionAccountService.optionAccountByUerId(userId); + + switch (req.getTransferOut()) { + case 1: + if (assets.getBalance().compareTo(req.getAmountMoney()) == -1) { + return R.error("转账失败,您的银行账户可用余额不足以本次转账"); + } + break; + case 2: + if (account.getBalance().compareTo(req.getAmountMoney()) == -1) { + return R.error("转账失败,您的期权期货账户可用余额不足以本次转账"); + } + break; + } + + if (service.modify(userId, req.getTransferOut(), req.getAmountMoney())) { + return R.ok("转账成功"); + } else { + return R.error(); + } + + } + + + /** + * @描述: 根据选择的转出账户查询可用资金 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-15 + **/ + @PostMapping(value = "/getCapital") + @ApiOperation(value = "根据选择的账户查询资金", response = CdTransferOutResp.class) + public R getCapital(@RequestParam("transferOut") @ApiParam(value = "账户类型(1.银行账户 2.期货账户)") Integer transferOut) { + int userId = TokenUtil.getUserId(); + + if (transferOut == 2) { + //期权账户 + CdUserOptionAccountEntity account = optionAccountService.optionAccountByUerId(Long.valueOf(userId)); + return R.ok().put("data", account); + } else if (transferOut == 1) { + //银行账户 + CdUserAssetsEntity assets = assetsService.userAssetsByUerId(Long.valueOf(userId)); + return R.ok().put("data", assets); + } + return R.error(); + + } + + + /** + * @描述: 转账记录 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-10 + **/ + @PostMapping(value = "/transferRecords") + @ApiOperation(value = "转账记录", response = CdTransferRecordsResp.class) + public R listByEntity(@RequestBody CdTransferRecordsReq req) { + int userId = TokenUtil.getUserId(); + + + List list = service.queryPageByRecode(String.valueOf(userId), req.getStartTime(), req.getEndTime()); + //总记录数 + int count = list.size(); + PageUtils pageUtils = new PageUtils(list, count, req.getPageSize(), req.getPageNum()); + + return R.ok().put("page", pageUtils); + + } + + + /** + * 期货期权账户 + * 同时此处会返回待开仓的信息 + * 可用资金 = 当前账户可用资金 + * + * @return + */ + @PostMapping(value = "/getAccountInfo") + @ApiOperation(value = "期货期权账户(此处会返回待开仓的信息)", response = CdAccountInfoResp.class) + public R getAccountInfo(@RequestBody CdGetAccountInfoReq req) { + + int userId = TokenUtil.getUserId(); + R r = service.getAccountInfoReturnOpen(req, userId); + /*//查询当前账户金额 + CdUserOptionAccountEntity entity = optionAccountService.optionAccountByUerId(userId); + + CdAccountInfoResp accountInfoResp = new CdAccountInfoResp(); + //可用资金 + accountInfoResp.setBalance(entity.getBalance()); + //冻结资金 + accountInfoResp.setForzenBalance(entity.getForzenBalance()); + //占用保证金 + accountInfoResp.setDepositBalance(entity.getDepositBalance()); + + //每手交易单位(合约面值) + Integer tradingUnit = 0; + + //查询当前用户的开仓信息 + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", userId); + + BigDecimal float_pl = new BigDecimal(0); + BigDecimal marketAssets = new BigDecimal(0); + //查询当前用户持仓信息 + CdFuturesConfigureEntity configureEntity = configureService.getOne(queryWrapper); + if (configureEntity != null) { + + //根据pid查询当前代码 + QhscGangjiaosuoEntity qhscGangjiaosuo = gangjiaosuoService.getById(configureEntity.getPid()); + + //根据代码查询当天汇率价格 + QueryWrapper wrapper = new QueryWrapper(); + wrapper.eq("e_name", qhscGangjiaosuo.getEName()); + wrapper.last(" AND TO_DAYS( NOW( ) ) - TO_DAYS( date) <= 1 "); + QhscGangjiaosuoEntity gangjiaosuo = gangjiaosuoService.getOne(wrapper); + + tradingUnit = CdFuturesConfigureServiceImpl.setCompany(gangjiaosuo.getName()); + + //浮动盈亏 =(当天结算价—昨日结算价)* 委托数量 * 合约面值 + float_pl = algorithm.getFloatingPL(gangjiaosuo.getLastPrice(), gangjiaosuo.getDayUp(), new BigDecimal(tradingUnit), configureEntity.getEntrustNumber()); + accountInfoResp.setFloatingPL(float_pl); + //资产市值 市值=外汇期货合约价格 * 委托数量 * 合约面值 + + //查询持仓信息状态是开仓还是平仓(0.开仓 1.平仓) + if (configureEntity.getStatus() == 0) { + + //资产市值 = (开仓或者平仓) * 委托数量 * 合约面值(1份合约=一手,一手=?吨) + marketAssets = algorithm.getMarketValue(configureEntity.getEntrustPrice(), configureEntity.getEntrustNumber(), tradingUnit); + + } else if (configureEntity.getStatus() == 1) { + marketAssets = algorithm.getMarketValue(configureEntity.getCloseRate(), configureEntity.getEntrustNumber(), tradingUnit); + } + + accountInfoResp.setMarketAssets(marketAssets); + } else { + accountInfoResp.setMarketAssets(marketAssets); + accountInfoResp.setFloatingPL(float_pl); + } + + + //总资金 = 资产市值 + 占用保证金 + 冻结资金 + 可用资金 + BigDecimal totalAssets = marketAssets.add(accountInfoResp.getDepositBalance().add(accountInfoResp.getBalance().add(accountInfoResp.getForzenBalance()))); + accountInfoResp.setTotalAssets(totalAssets); + + + QhscGangjiaosuoEntity qh = gangjiaosuoService.getById(req.getId()); + + //序号 + accountInfoResp.setId(req.getId()); + //代码 + accountInfoResp.setEname(qh.getEName()); + //合约名称 + accountInfoResp.setName(qh.getName()); + //交易方向 + accountInfoResp.setTradingDirection(req.getTradingDirection()); + //需要资金 + + int unit = CdFuturesConfigureServiceImpl.setCompany(qh.getName()); + + //所需资金 = 委托价格 * 委托数量(手数) * 每手交易单位 + BigDecimal totalAmount = algorithm.getFundsRequired(req.getEntrustPrice(), BigDecimal.valueOf(req.getEntrustNumber()), unit); + accountInfoResp.setFundingNeeded(totalAmount); + + //创建时间 + accountInfoResp.setCreateTime(DateUtils.getNowTime()); + //最后交易日 + //获取最后交易日 + String tradeTime = algorithm.getLastTradingDay(qh.getName()); + accountInfoResp.setTradeTime(tradeTime);*/ + + return r; + + } + + + /** + * 期货期权账户 + * + * @return + */ + @PostMapping(value = "/getAccount") + @ApiOperation(value = "期货期权账户", response = CdAccountInfoResp.class) + public R getAccountInfo() { + + int userId = TokenUtil.getUserId(); + String getDate = template.opsForValue().get("date:userId_" + userId); + + //查询当前账户金额 + CdUserOptionAccountEntity entity = optionAccountService.optionAccountByUerId(userId); + + CdAccountInfoResp accountInfoResp = new CdAccountInfoResp(); + //可用资金 + accountInfoResp.setBalance(entity.getBalance()); + //冻结资金 + accountInfoResp.setForzenBalance(entity.getForzenBalance()); + //占用保证金 + accountInfoResp.setDepositBalance(entity.getDepositBalance()); + + + //查询当前用户的开仓信息 + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", userId); + + BigDecimal float_pl = new BigDecimal(0); + BigDecimal marketAssets = new BigDecimal(0); + //查询当前用户持仓信息 + CdFuturesConfigureEntity configureEntity = configureService.getOne(queryWrapper); + if (configureEntity != null) { + + //根据pid查询当前代码 + QhscGangjiaosuoEntity qhscGangjiaosuo = gangjiaosuoService.getById(configureEntity.getPid()); + + //根据代码查询当天汇率价格 + QueryWrapper wrapper = new QueryWrapper(); + wrapper.eq("e_name", qhscGangjiaosuo.getEName()); + wrapper.eq("date", getDate); + QhscGangjiaosuoEntity gangjiaosuo = gangjiaosuoService.getOne(wrapper); + + //每手交易单位(合约面值) + Integer tradingUnit = CdFuturesConfigureServiceImpl.setCompany(gangjiaosuo.getName()); + + //浮动盈亏 =(当天结算价—昨日结算价)* 委托数量 * 合约面值 + float_pl = algorithm.getFloatingPL(gangjiaosuo.getLastPrice(), gangjiaosuo.getDayUp(), new BigDecimal(tradingUnit), configureEntity.getEntrustNumber()); + accountInfoResp.setFloatingPL(float_pl); + //资产市值 市值=外汇期货合约价格 * 委托数量 * 合约面值 + + //查询持仓信息状态是开仓还是平仓(0.开仓 1.平仓) + if (configureEntity.getStatus() == 0) { + + //资产市值 = (开仓或者平仓) * 委托数量 * 合约面值(1份合约=一手,一手=?吨) + marketAssets = algorithm.getMarketValue(configureEntity.getEntrustPrice(), configureEntity.getEntrustNumber(), tradingUnit); + + } else if (configureEntity.getStatus() == 1) { + marketAssets = algorithm.getMarketValue(configureEntity.getCloseRate(), configureEntity.getEntrustNumber(), tradingUnit); + } + + accountInfoResp.setMarketAssets(marketAssets); + } else { + accountInfoResp.setMarketAssets(marketAssets); + accountInfoResp.setFloatingPL(float_pl); + } + + + //总资金 = 资产市值 + 占用保证金 + 冻结资金 + 可用资金 + BigDecimal totalAssets = marketAssets.add(accountInfoResp.getDepositBalance().add(accountInfoResp.getBalance().add(accountInfoResp.getForzenBalance()))); + accountInfoResp.setTotalAssets(totalAssets); + + + return R.ok().put("data", accountInfoResp); + } +} + diff --git a/src/main/java/com/msdw/tms/controller/CdCacheController.java b/src/main/java/com/msdw/tms/controller/CdCacheController.java new file mode 100644 index 0000000..df221d1 --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/CdCacheController.java @@ -0,0 +1,84 @@ +package com.msdw.tms.controller; + +import com.alibaba.fastjson.JSONObject; +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.TokenUtil; +import com.msdw.tms.entity.CdCacheEntity; +import com.msdw.tms.model.req.CdSaveCacheReq; +import com.msdw.tms.service.CdCacheService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiOperation; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + + +/** + * @描述:步骤缓存记录表控制类 + * @作者: Rong + * @日期: 2020-10-21 + */ + +@RestController +@RequestMapping("/chuanda/cache") +@Api(value = "API - 川大:步骤记录", tags = "川大子系统:步骤记录") +public class CdCacheController { + + @Autowired + public CdCacheService service; + + + /** + * @描述: 新增步骤缓存 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-21 + **/ + @PostMapping("/save") + @ApiOperation(value = "步骤缓存", response = CdSaveCacheReq.class) + public R save(@RequestBody CdSaveCacheReq req) { + int userId = TokenUtil.getUserId(); + return service.cache(userId, req.getStepNumber(), req.getJson()); + + } + + /** + * @描述: 根据用户id查询步骤 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-21 + **/ + @PostMapping("/list") + @ApiOperation(value = "查询步骤/实验报告", response = CdCacheEntity.class) + public R list() { + int userId = TokenUtil.getUserId(); + QueryWrapper wrapper = new QueryWrapper<>(); + wrapper.eq("user_id", userId); + CdCacheEntity cache = service.getOne(wrapper); + + + cache.getStep1(); + return R.ok().put("data", cache); + + } + + + public static void main(String[] args) { + javaToJSON(); + } + + public static void javaToJSON() { + JSONObject jsonObject = new JSONObject(); + + jsonObject.put("userId", "1"); + System.out.println("java--->json \n " + jsonObject.toString()); + } + + +} + diff --git a/src/main/java/com/msdw/tms/controller/CdExperimentalController.java b/src/main/java/com/msdw/tms/controller/CdExperimentalController.java new file mode 100644 index 0000000..bcceda6 --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/CdExperimentalController.java @@ -0,0 +1,197 @@ +package com.msdw.tms.controller; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.TokenUtil; +import com.msdw.tms.entity.CdExperimentalEntity; +import com.msdw.tms.model.req.CdExperimentReq; +import com.msdw.tms.service.CdExperimentalService; +import io.swagger.annotations.*; +import org.apache.ibatis.annotations.Param; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.redis.core.RedisTemplate; +import org.springframework.http.MediaType; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletResponse; +import java.util.List; + + +/** + * @描述:实验数据控制类 + * @作者: Rong + * @日期: 2020-10-20 + */ +@RestController +@RequestMapping("/chuanda/experimental") +@Api(value = "API - 川大:实验数据", tags = "川大子系统:实验数据") +public class CdExperimentalController { + + @Autowired + public CdExperimentalService service; + + @Autowired + public RedisTemplate template; + + /** + * @描述: 实验数据展示 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-20 + **/ + @PostMapping("/listByCondition") + @ApiOperation(value = " 实验数据展示", response = CdExperimentalEntity.class) + public R listByEntity(@RequestParam("type") @ApiParam(value = "实验类型(0正态Var,1历史模拟,2蒙特卡洛)") Integer type) { + + int userId = TokenUtil.getUserId(); + QueryWrapper entity = new QueryWrapper<>(); + entity.eq("type", type); + entity.eq("user_id", userId); + List list = service.list(entity); + return R.ok().put("data", list); + } + + + /** + * @描述: 保存实验 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-09-28 + **/ + @ApiOperation(value = "保存实验") + @PostMapping(value = "/save", produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + @CrossOrigin + public R save(@RequestBody CdExperimentReq req) { + int userId = TokenUtil.getUserId(); + + CdExperimentalEntity entity = new CdExperimentalEntity(); + String getDate = template.opsForValue().get("date:userId_" + userId); + //当前时间 + entity.setCreatTime(getDate); + //汇率类型 + entity.setType(Integer.valueOf(req.getType())); + //汇率名称 + entity.setExchangeRate(req.getName()); + //样本数据 + entity.setSampleData(req.getSampleData()); + //置信水平 + entity.setConfidenceLevel(req.getConfidenceLevel()); + //持有天数 + entity.setValidPeriod(req.getValidPeriod()); + + entity.setUserId(Integer.valueOf(userId)); + + //实验类型(0正态Var,1历史模拟,2蒙特卡洛) + switch (req.getType()) { + case "0": + case "1": + //Var值 + entity.setVarCode(req.getVarCode()); + break; + + case "2": + //实验次数 + entity.setExperimentsNumber(Integer.valueOf(req.getExperimentsNumber())); + //汇率价格走势图 + entity.setTrendChart(req.getTrendChart()); + //汇率损益分布图 + entity.setProfitLossDistribution(req.getProfitLossDistribution()); + break; + } + boolean result = service.save(entity); + if (result) { + return R.ok(); + } + return R.error(); + } + + + /** + * @描述: 根据主键删除 + * @入参: id + * @出参: + * @作者: Rong + * @日期: 2020-10-20 + **/ + @PostMapping("/delete") + @ApiOperation(value = "根据主键删除") + public R delete(@RequestParam("id") @ApiParam(value = "要删除的id") Integer id) { + service.removeById(id); + return R.ok(); + } + + + /** + * @描述: 批量删除 + * @入参: id + * @出参: + * @作者: Rong + * @日期: 2020-10-20 + **/ + @PostMapping("/deleteBatchIds") + @ApiOperation(value = "批量删除") + public R deleteBatchIds(@RequestParam("ids") @ApiParam(value = "要删除的id列表") List ids) { + boolean result = service.removeByIds(ids); + if (result) { + return R.ok(); + } + return R.error(); + } + + + /** + * @描述: 根据时间、币种条件查询后导出报表 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-09-28 + **/ + @ApiOperation(value = "单条导出实验结果") + @GetMapping(value = "/export", produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + @ResponseBody + @CrossOrigin + @ApiImplicitParams({ + @ApiImplicitParam(name = "id", value = "序号", required = true, dataType = "String", paramType = "query") + }) + public R export(HttpServletResponse response, @Param(value = "id") Integer id) { + + try { + //TODO:历史数据下载:只能选取所选择的行情日期之前的数据下载 + R r = service.exportProjectRecord(response, id); + + return r; + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + + } + + /** + * 根据类型导出相应实验结果 + * + * @param response + * @param ids + * @return + */ + @ApiOperation(value = "根据类型导出相应实验结果", response = CdExperimentalEntity.class) + @GetMapping(value = "/exportByAll", produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + @ResponseBody + @CrossOrigin + public R exportByAll(HttpServletResponse response, @RequestParam("ids") @ApiParam(value = "要导出的id") List ids) { + try { + int userId = TokenUtil.getUserId(); + R r = service.exportexportProjectRecordById(response, userId, ids); + return r; + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + + } + + +} + diff --git a/src/main/java/com/msdw/tms/controller/CdFuturesConfigureController.java b/src/main/java/com/msdw/tms/controller/CdFuturesConfigureController.java new file mode 100644 index 0000000..9e3a77f --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/CdFuturesConfigureController.java @@ -0,0 +1,189 @@ +package com.msdw.tms.controller; + +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.TokenUtil; +import com.msdw.tms.entity.CdFuturesConfigureEntity; +import com.msdw.tms.entity.vo.CdFuturesConfigureEntityVo; +import com.msdw.tms.model.req.CdBuyInReq; +import com.msdw.tms.model.req.CdOpenPositionReq; +import com.msdw.tms.model.resp.CdAfterHedgingResp; +import com.msdw.tms.model.resp.CdBeforeHedgingResp; +import com.msdw.tms.model.resp.CdHoldPositionInfoResp; +import com.msdw.tms.model.resp.CdTransferWarehouseDetailResp; +import com.msdw.tms.service.CdFuturesConfigureService; +import com.msdw.tms.service.QhscGangjiaosuoService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiOperation; +import io.swagger.annotations.ApiParam; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.*; + + +/** + * @描述:期货资产头寸配置控制类 + * @作者: Rong + * @日期: 2020-10-10 + */ +@RestController +@RequestMapping("/chuanda/futuresConfigure") +@Api(tags = "川大子系统:开仓平仓相关操作") +public class CdFuturesConfigureController { + + @Autowired + public CdFuturesConfigureService service; + + @Autowired + public QhscGangjiaosuoService qhscGangjiaosuoService; + + /** + * @描述: 开仓 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-10 + **/ + @PostMapping("/save") + @ApiOperation(value = "开仓操作") + public R save(@RequestBody CdOpenPositionReq req) { + int userId = TokenUtil.getUserId(); + + if (req.getEntrustNumber() <= 0) { + return R.error("委托数量不能小于等于0!"); + } + CdFuturesConfigureEntity entity = new CdFuturesConfigureEntity(); + entity.setUserId(userId); + entity.setPid(req.getPid()); + entity.setTradingDirection(req.getTradingDirection()); + entity.setNewestPrice(req.getNewestPrice()); + entity.setEntrustPrice(req.getEntrustPrice()); + entity.setEntrustNumber(req.getEntrustNumber()); + entity.setFundingNeeded(req.getFundingNeeded()); + + + return service.openPosition(entity); + } + + /** + * @描述: 外汇期货持仓情况 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-12 + **/ + @PostMapping("/findById") + @ApiOperation(value = "外汇期货持仓情况", response = CdHoldPositionInfoResp.class) + public R findById() { + try { + int userId = TokenUtil.getUserId(); + CdHoldPositionInfoResp entity = service.myPosition(userId); + + if (entity == null) { + return R.error("暂无持仓信息"); + } + return R.ok().put("data", entity); + + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + } + + /** + * @描述: 平仓 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-10 + **/ + @PostMapping("/buyIn") + @ApiOperation(value = "平仓操作", response = CdFuturesConfigureEntity.class) + public R closePosition(@RequestBody CdBuyInReq req) { + int userId = TokenUtil.getUserId(); + return service.buyIn(req.getId(), req.getDate(), userId); + + + } + + /** + * @描述: 调仓详情 + * @作者: Rong + * @日期: 2020-10-14 + **/ + @PostMapping("/detailByTransferWarehouse") + @ApiOperation(value = "调仓详情", response = CdTransferWarehouseDetailResp.class) + public R transferWarehouse(@RequestParam("id") @ApiParam(value = "序号") Integer id) { + try { + CdTransferWarehouseDetailResp configure = service.transferWarehouseByDetail(id); + + + return R.ok().put("data", configure); + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + + + } + + /** + * @描述: 外汇期货平仓绩效 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-10 + **/ + @PostMapping("/buyInAchievements") + @ApiOperation(value = "外汇期货平仓绩效", response = CdFuturesConfigureEntityVo.class) + public R buyInAchievements(@RequestParam("id") @ApiParam(value = "序号") Integer id) { + try { + int userId = TokenUtil.getUserId(); + CdFuturesConfigureEntityVo entity = service.buyInAchievements(id, userId); + return R.ok().put("data", entity); + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + + } + + /** + * 对冲前-外汇现货盈亏 + * + * @return + */ + @PostMapping("/beforeHedging") + @ApiOperation(value = "对冲前-外汇现货盈亏", response = CdBeforeHedgingResp.class) + public R beforeHedging() { + try { + int userId = TokenUtil.getUserId(); + CdBeforeHedgingResp cdBeforeHedgingResp = service.beforeHedging(userId); + return R.ok().put("data", cdBeforeHedgingResp); + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + + } + + /** + * 对冲后损益情况 + * 样本区间:取开仓时间-开仓后的三个月 + * + * @return + */ + @PostMapping("/afterHedging") + @ApiOperation(value = "对冲后损益情况", response = CdAfterHedgingResp.class) + public R afterHedging() { + try { + int userId = TokenUtil.getUserId(); + CdAfterHedgingResp afterHedging = service.afterHedging(userId); + return R.ok().put("data", afterHedging); + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + + } + +} + diff --git a/src/main/java/com/msdw/tms/controller/CdHedgingController.java b/src/main/java/com/msdw/tms/controller/CdHedgingController.java new file mode 100644 index 0000000..248bf57 --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/CdHedgingController.java @@ -0,0 +1,193 @@ +package com.msdw.tms.controller; + +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.TokenUtil; +import com.msdw.tms.common.utils.date.DateUtils; +import com.msdw.tms.entity.CdFuturesConfigureEntity; +import com.msdw.tms.entity.vo.CdFuturesConfigureEntityVo; +import com.msdw.tms.model.resp.CdBeforeHedgingResp; +import com.msdw.tms.model.resp.CdHedgingEfficiencyResp; +import com.msdw.tms.model.resp.CdHedgingResp; +import com.msdw.tms.service.CdFuturesConfigureService; +import com.msdw.tms.service.impl.CdFuturesConfigureServiceImpl; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiOperation; +import io.swagger.annotations.ApiParam; +import org.springframework.beans.factory.annotation.Autowired; +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 java.math.BigDecimal; +import java.math.RoundingMode; +import java.text.DecimalFormat; + + +/** + * @描述:对冲 + * @作者: Rong + * @日期: 2020-10-21 + */ + +@RestController +@RequestMapping("/chuanda/hedging") +@Api(value = "API - 川大:对冲", tags = "川大子系统:对冲") +public class CdHedgingController { + @Autowired + public CdFuturesConfigureService configureService; + + /** + * @描述: 对冲 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-21 + **/ + @PostMapping("/hedgingInfo") + @ApiOperation(value = "风险对冲核算", response = CdHedgingResp.class) + public R buyInAchievements(@RequestParam("id") @ApiParam(value = "序号") Integer id) { + try { + + CdHedgingResp resp = new CdHedgingResp(); + int userId = TokenUtil.getUserId(); + + //转换百分比 + DecimalFormat df = new DecimalFormat("0.00%"); + + //期货开仓平仓记录 + CdFuturesConfigureEntity entity = configureService.getById(id); + + CdFuturesConfigureEntityVo configure = configureService.buyInAchievements(id, userId); + configure.setEntrustPrice(entity.getEntrustPrice()); + + CdBeforeHedgingResp beforeHedging = configureService.beforeHedging(userId); + String getCreateTime = DateUtils.dateForByStr(beforeHedging.getCreateTime()); + String getUpdateTime = DateUtils.dateForByStr(beforeHedging.getUpdateTime()); + + beforeHedging.setCreateTime(getCreateTime); + beforeHedging.setUpdateTime(getUpdateTime); + + + CdHedgingEfficiencyResp efficiencyResp = new CdHedgingEfficiencyResp(); + + + //获取期货合约面值 + Integer setCompany = CdFuturesConfigureServiceImpl.setCompany(configure.getName()); + + + //合约名称 + efficiencyResp.setName(configure.getName().substring(0, 6)); + + + //现货盈亏 + efficiencyResp.setPromptGoodsPL(beforeHedging.getProfitLoss()); + + //期货盈亏 + efficiencyResp.setFuturesPl(configure.getWh_pl()); + + + //总盈亏 = 外汇期货盈亏 + 外汇现货盈亏 + BigDecimal totalPl = configure.getWh_pl().add(beforeHedging.getProfitLoss()); + efficiencyResp.setTotalPl(totalPl); + //totalPl = totalPl.divide(new BigDecimal(3), 2, BigDecimal.ROUND_HALF_UP); + + //外汇期货盈亏 + Integer getWh_pl = configure.getWh_pl().intValue(); + + //外汇现货盈亏的绝对值 + Integer getProfitLoss = beforeHedging.getProfitLoss().intValue(); + + + //风险对冲效率(%) = 外汇期货盈亏 / 外汇现货盈亏的绝对值 + double getWh_plRiskEfficiency = getAbsolute(getWh_pl); + double getProfitLossRiskEfficiency = getAbsolute(getProfitLoss); + + + BigDecimal RiskEfficiency = new BigDecimal(getProfitLossRiskEfficiency); + if (RiskEfficiency.intValue() == 0) { + efficiencyResp.setRiskEfficiency((0) + "%"); + } else { + BigDecimal riskEfficiency = new BigDecimal(getWh_plRiskEfficiency).divide(RiskEfficiency, 4, RoundingMode.HALF_UP); + efficiencyResp.setRiskEfficiency((riskEfficiency.multiply(new BigDecimal(100))) + "%"); + } + + //风险对冲比率(%)=(期货合约面值 * 持仓总量) / 外汇金额 + + double num = configure.getEntrustNumber(); + + //期货合约面值 * 持仓总量 + double price = setCompany * num; + + /** + * 如果 风险对冲比率(%)=(期货合约面值 * 持仓总量) /所需要的的金额=(期货合约价*持仓总量*每手) + * 比如合约每手价格为6.85元,合约面值为 100000 持仓量为1手 + * (100000 * 1)/(6.85*100000 *1) + */ + + + //TODO 风险对冲比率(%)=(期货合约面值 * 持仓总量) / 外汇金额(10000000元) + //外汇金额=所需资金 + //BigDecimal needPrice = BigDecimal.valueOf(setCompany).multiply(BigDecimal.valueOf(configure.getEntrustNumber())).multiply(configure.getEntrustPrice()); + // BigDecimal riskRatio = price.subtract(needPrice); + + double riskRatio = price / 10000000; + + + //double percentage2 = riskRatio / setNum; + + //riskRatio转换为% + efficiencyResp.setRiskRatio(riskRatio + "%"); + + BigDecimal need = beforeHedging.getNeededPriceByOpen().add(beforeHedging.getNeededPriceByClose()); + + //need = need.divide(new BigDecimal(3), 2, BigDecimal.ROUND_HALF_UP); + + //总盈亏率 = (外汇期货盈亏+外汇现货的盈亏)/(外汇期货所需资金+外汇现货所需资金) + BigDecimal totalPlRatio = totalPl.divide(need, 4, RoundingMode.HALF_UP); + efficiencyResp.setTotalPlRatio(totalPlRatio.multiply(new BigDecimal(100)) + "%"); + + + //外汇期货平仓绩效 + resp.setConfigure(configure); + //外汇现货盈亏 + resp.setBeforeHedging(beforeHedging); + //对冲效率 + resp.setHedgingEfficiency(efficiencyResp); + + return R.ok().put("data", resp); + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + + } + + //获取绝对值 + public static int getAbsolute(Integer num) { + return num * (1 - ((num >>> 31) << 1)); + } + + public static void main(String[] args) { + // System.out.println(getAbsolute(5)); + + //转换百分比 + DecimalFormat df = new DecimalFormat("0.00%"); + //System.out.println(df.format(0.00196778)); + + double a = 1000000; + double b = 10000000; + + double c = a / b; + System.out.println(c); + + + } + + /* public static void main(String[] args) { + String str = "日元兑人民币期货2010"; + System.out.println(str.substring(0,2)); + }*/ + +} + diff --git a/src/main/java/com/msdw/tms/controller/CdPointController.java b/src/main/java/com/msdw/tms/controller/CdPointController.java new file mode 100644 index 0000000..c4c3a5f --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/CdPointController.java @@ -0,0 +1,88 @@ +package com.msdw.tms.controller; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.entity.CdPoint; +import com.msdw.tms.entity.CdPointChildren; +import com.msdw.tms.service.CdPointChildrenService; +import com.msdw.tms.service.CdPointService; +import io.swagger.annotations.*; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.*; + +import java.util.List; + + +/** + * @描述:控制类 + * @作者: Rong + * @日期: 2021-01-11 + */ +@RestController +@RequestMapping("/point") +@Api(value = "API - 判分点:CdPointController", tags = "判分点:") +public class CdPointController { + + @Autowired + public CdPointService service; + + @Autowired + public CdPointChildrenService childrenService; + + /** + * @描述: 查询一级目录 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2021-01-11 + **/ + @CrossOrigin + @GetMapping("/firstLevel") + @ApiOperation(value = " 查询一级目录", response = CdPoint.class) + public R listByEntity() { + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("parent_id", 0); + + List pointsList = service.list(queryWrapper); + + return R.ok().put("list", pointsList); + } + + + /** + * @描述: 查询二级目录 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2021-01-11 + **/ + @CrossOrigin + @GetMapping("/secondaryLevel") + @ApiOperation(value = " 查询二级目录", response = CdPoint.class) + public R secondaryLevel(@RequestParam("id") @ApiParam(value = "序号") Integer id) { + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("parent_id", id); + List pointsList = service.list(queryWrapper); + return R.ok().put("list", pointsList); + } + + + /** + * @描述: 查询三级目录 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2021-01-11 + **/ + @CrossOrigin + @GetMapping("/thirdLevel") + @ApiOperation(value = " 查询三级目录", response = CdPointChildren.class) + public R thirdLevel(@RequestParam("id") @ApiParam(value = "序号") Integer id) { + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("point_id", id); + List pointsList = childrenService.list(queryWrapper); + return R.ok().put("list", pointsList); + } + +} + diff --git a/src/main/java/com/msdw/tms/controller/CdTransferWarehouseRecordController.java b/src/main/java/com/msdw/tms/controller/CdTransferWarehouseRecordController.java new file mode 100644 index 0000000..536df89 --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/CdTransferWarehouseRecordController.java @@ -0,0 +1,63 @@ +package com.msdw.tms.controller; + +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.TokenUtil; +import com.msdw.tms.entity.CdFuturesConfigureEntity; +import com.msdw.tms.model.req.CdTransferWarehouseReq; +import com.msdw.tms.service.CdTransferWarehouseRecordService; +import com.msdw.tms.service.CdUserAssetsService; +import com.msdw.tms.service.CdUserOptionAccountService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiOperation; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + + +/** + * @描述:调仓记录表控制类 + * @作者: Rong + * @日期: 2020-10-10 + */ +@RestController +@RequestMapping("/chuanda/transferWarehouseRecord") +@Api(value = "API - 川大:调仓相关操作", tags = "川大子系统:调仓相关操作") +public class CdTransferWarehouseRecordController { + + @Autowired + public CdTransferWarehouseRecordService service; + + @Autowired + public CdUserOptionAccountService optionAccountService; + + @Autowired + public CdUserAssetsService assetsService; + + /** + * @描述: 调仓操作 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-10 + **/ + @PostMapping("/modify") + @ApiOperation(value = "调仓操作") + public R modify(@RequestBody CdTransferWarehouseReq req) { + if (req.getEntrustNumber() <= 0) { + return R.error("委托数量不能小于等于0!"); + } + CdFuturesConfigureEntity entity = new CdFuturesConfigureEntity(); + entity.setUserId(TokenUtil.getUserId()); + entity.setId(req.getId()); + /*entity.setTradingDirection(req.getTradingDirection());*/ + entity.setStatus(req.getStatus()); + entity.setEntrustPrice(req.getEntrustPrice()); + entity.setEntrustNumber(req.getEntrustNumber()); + entity.setTradableNumber(req.getTradableNumber()); + return service.add(entity); + } + +} + diff --git a/src/main/java/com/msdw/tms/controller/CdUserAssetsController.java b/src/main/java/com/msdw/tms/controller/CdUserAssetsController.java new file mode 100644 index 0000000..32a1bbf --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/CdUserAssetsController.java @@ -0,0 +1,87 @@ +package com.msdw.tms.controller; + +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.date.DateUtils; +import com.msdw.tms.entity.CdUserAssetsEntity; +import com.msdw.tms.entity.CdUserOptionAccountEntity; +import com.msdw.tms.service.CdUserAssetsService; +import com.msdw.tms.service.CdUserOptionAccountService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiOperation; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.redis.core.RedisTemplate; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + +import javax.annotation.Resource; +import java.math.BigDecimal; +import java.time.LocalDateTime; + + +/** + * @描述:用户银行账户控制类 + * @作者: Rong + * @日期: 2020-10-10 + */ +@RestController +@RequestMapping("/chuanda/userAssets") +@Api(value = "API - 川大:账户信息新增", tags = "川大子系统:账户信息新增") +public class CdUserAssetsController { + + @Autowired + public CdUserAssetsService service; + + @Autowired + public CdUserOptionAccountService optionAccountService; + + @Resource + public RedisTemplate template; + + /** + * @描述: 新增银行账户信息和期权账户信息 + * @作者: Rong + * @日期: 2020-10-10 + **/ + @PostMapping("/save") + @ApiOperation(value = "新增银行账户信息和期权账户信息") + public R save(Integer userId) { + + try { + + String getDate = template.opsForValue().get("date:userId_" + userId); + //新增银行信息 + CdUserAssetsEntity userAssets = new CdUserAssetsEntity(Long.valueOf(userId), new BigDecimal(100000), new BigDecimal(100000), getDate); + + //新增期权账户信息 + CdUserOptionAccountEntity optionAccount = new CdUserOptionAccountEntity(Long.valueOf(userId), new BigDecimal(100000), new BigDecimal(0), new BigDecimal(0), getDate); + + //TODO 判断期权表和银行账户有没有过userId,没有则新增 + + //1.银行账户信息 + CdUserAssetsEntity getBankInfo = service.userAssetsByUerId(userId); + CdUserOptionAccountEntity getOptionInfo = optionAccountService.optionAccountByUerId(userId); + if (getBankInfo == null && getOptionInfo == null) { + //TODO getDate:这里是获取步骤二选中的时间(历史时间) + userAssets.setCreateTime(getDate); + userAssets.setUpdateTime(DateUtils.formatTime((LocalDateTime.now()))); + + //设置当前时间 + optionAccount.setCreateTime(getDate); + optionAccount.setUpdateTime(DateUtils.formatTime((LocalDateTime.now()))); + + service.addInfo(userAssets); + optionAccountService.addInfoByOptionAccount(optionAccount); + } + return R.ok("新增账户信息成功"); + } catch (Exception e) { + e.printStackTrace(); + return R.error("账户信息新增失败"); + } + + + } + + +} + diff --git a/src/main/java/com/msdw/tms/controller/QhscGangjiaosuoController.java b/src/main/java/com/msdw/tms/controller/QhscGangjiaosuoController.java new file mode 100644 index 0000000..f06b6f9 --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/QhscGangjiaosuoController.java @@ -0,0 +1,209 @@ +package com.msdw.tms.controller; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.msdw.tms.common.utils.PageUtils; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.TokenUtil; +import com.msdw.tms.common.utils.date.DateUtils; +import com.msdw.tms.common.utils.trading_rules.algorithm; +import com.msdw.tms.entity.QhscGangjiaosuoEntity; +import com.msdw.tms.entity.WhscRenminbipinzhongEntity; +import com.msdw.tms.model.req.QhscGangjiaosuoReq; +import com.msdw.tms.model.resp.CdGetDetailByIdResp; +import com.msdw.tms.model.resp.CdGetExchangeRateResp; +import com.msdw.tms.service.QhscGangjiaosuoService; +import com.msdw.tms.service.WhscRenminbipinzhongService; +import com.msdw.tms.service.impl.CdFuturesConfigureServiceImpl; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiOperation; +import io.swagger.annotations.ApiParam; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.*; + +import java.math.BigDecimal; +import java.util.*; + + +/** + * @描述:控制类 + * @作者: Rong + * @日期: 2020-10-10 + */ +@RestController +@RequestMapping("/chuanda/qhscGangjiaosuo") +@Api(value = "API - 川大:期货合约", tags = "川大子系统:期货合约") +public class QhscGangjiaosuoController { + + @Autowired + public QhscGangjiaosuoService service; + + @Autowired + public WhscRenminbipinzhongService rmbService; + + /** + * @描述: 根据实体属性分页查询 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-10-10 + **/ + @PostMapping("/listByEntity") + @ApiOperation(value = " 列表:根据外币币种分页查询", response = QhscGangjiaosuoEntity.class) + public R listByEntity(@RequestBody QhscGangjiaosuoReq req) { + + Integer userId = TokenUtil.getUserId(); + if (null == req.getDate()) { + return R.error("选择时间不能为空"); + } + PageUtils list = service.queryPageByQh(req.getPageNum(), req.getPageSize(), req.getName(), req.getDate().substring(0, 10)); + return R.ok().put("page", list); + } + + + /** + * 列表:查询所有外币币种类型 + * + * @return + */ + @GetMapping("/getAllCurrency") + @ApiOperation(value = " 列表:查询所有外币币种类型", response = QhscGangjiaosuoEntity.class) + public Map getAllCurrency() { + + Integer userId = TokenUtil.getUserId(); + List list = service.getName(); + + QueryWrapper wrapper = new QueryWrapper(); + wrapper.orderByAsc("date").last("limit 1"); + WhscRenminbipinzhongEntity date = rmbService.getOne(wrapper); + + + Iterator it = list.iterator(); + while (it.hasNext()) { + String x = it.next().getCurrency(); + if (x.equals("人民币")) { + it.remove(); + } + } + String time = DateUtils.dateForByStr(date.getDate()); + Map map = new HashMap<>(); + map.put("errmessage", "success"); + map.put("data", list); + map.put("status", 200); + map.put("startingTime", time);//获取外汇市场人民币品种最前一天的日期 + + + return map; + //return R.ok().put("data", list); + + } + + + /** + * 列表:汇率类型 + * 以中间表查询 + * + * @return + */ + @GetMapping("/getExchangeRate") + @ApiOperation(value = " 汇率类型", response = CdGetExchangeRateResp.class) + public R getExchangeRate() { + + Integer userId = TokenUtil.getUserId(); + //查询汇率类型 + List list = rmbService.getExchangeRate(); + + //respList:用来保存去除其他字段的list + List respList = new ArrayList<>(); + + //查询币种名称 + List getRMB_Name = service.getName(); + Iterator it = getRMB_Name.iterator(); + while (it.hasNext()) { + String x = it.next().getCurrency(); + if (x.equals("人民币")) { + it.remove(); + } + } + + for (WhscRenminbipinzhongEntity entity : list) { + CdGetExchangeRateResp rateResp = new CdGetExchangeRateResp(); + //name:循环出中间价的类型 + String name = entity.getName(); + + //循环出人民币币种 + for (QhscGangjiaosuoEntity g : getRMB_Name) { + String rmbType = g.getCurrency(); + //判断中间价 是否包含 人民币币种 + if (name.contains(rmbType)) { + rateResp.setTypeName(name); + } + } + + if (rateResp.getTypeName() != null) { + respList.add(rateResp); + } + + } + + + return R.ok().put("data", respList); + + } + + + /** + * 外汇期货合约类型选择—根据id查看配置头寸信息 + * + * @return + */ + @PostMapping("/getDetailById") + @ApiOperation(value = " 外汇期货合约类型选择—根据id查看配置头寸信息", response = CdGetDetailByIdResp.class) + public R getDetailById(@RequestParam("id") @ApiParam(value = "要查询的合约id") Integer id) { + + + Integer userId = TokenUtil.getUserId(); + + //根据id查询合约的信息 + QhscGangjiaosuoEntity exchange = this.service.getById(id); + + //设置每手交易单位(合约面值) + Integer tradingUnit = 0; + tradingUnit = CdFuturesConfigureServiceImpl.setCompany(exchange.getName()); + + CdGetDetailByIdResp resp = new CdGetDetailByIdResp(); + //最新价格 + resp.setNewestPrice(exchange.getLastPrice()); + //合约名称 + resp.setName(exchange.getName()); + //可交易数量 + resp.setTradableNumber(exchange.getHava()); + //委托价(暂为合约最新价) + resp.setEntrustPrice(exchange.getLastPrice()); + //委托数量(暂为1手) + resp.setEntrustNumber(1); + + + //:预冻结保证金 = 外汇期货合约价格 * 合约面值 * 手数 * 保证金比例 + BigDecimal getPreFreezingMargin = algorithm.getPreFreezingMargin(exchange.getLastPrice(), new BigDecimal(tradingUnit), exchange.getName().substring(0, 8)); + resp.setPreFreezingMargin(getPreFreezingMargin); + + //:预冻结手续费 = 委托数量(手)* 费用标准 + BigDecimal getPreFreezingServiceCharge = algorithm.getPreFreezingServiceCharge(new BigDecimal(resp.getEntrustNumber())); + resp.setPreFreezingFrozen(getPreFreezingServiceCharge); + + //:冻结资金 = 预计冻结保证金+预冻结手续费 + BigDecimal frozen_price = getPreFreezingMargin.add(getPreFreezingServiceCharge); + resp.setFrozenPrice(frozen_price); + resp.setPid(id); + resp.setTradingUnit(tradingUnit); + BigDecimal bond = algorithm.getBond(exchange.getName()); + resp.setTradingBond(bond); + + return R.ok().put("data", resp); + + + } + + +} + diff --git a/src/main/java/com/msdw/tms/controller/WhscRenminbipinzhongController.java b/src/main/java/com/msdw/tms/controller/WhscRenminbipinzhongController.java new file mode 100644 index 0000000..bfaec26 --- /dev/null +++ b/src/main/java/com/msdw/tms/controller/WhscRenminbipinzhongController.java @@ -0,0 +1,197 @@ +package com.msdw.tms.controller; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.msdw.tms.common.utils.PageUtils; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.TokenUtil; +import com.msdw.tms.entity.WhscRenminbipinzhongEntity; +import com.msdw.tms.entity.vo.CdCreatePhotoVo; +import com.msdw.tms.model.req.CdExportReq; +import com.msdw.tms.model.req.CdGetWeekAndMonthInfoReq; +import com.msdw.tms.model.req.CdPageConditionReq; +import com.msdw.tms.model.req.CdSimulationReq; +import com.msdw.tms.model.resp.CdGetWeekAndMonthInfoResp; +import com.msdw.tms.service.WhscRenminbipinzhongService; +import io.swagger.annotations.Api; +import io.swagger.annotations.ApiOperation; +import io.swagger.annotations.ApiParam; +import org.springframework.http.MediaType; +import org.springframework.web.bind.annotation.*; +import redis.clients.jedis.Jedis; + +import javax.annotation.Resource; +import javax.servlet.http.HttpServletResponse; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + + +/** + * @描述:外汇市场人民币品种控制类 + * @作者: Rong + * @日期: 2020-09-28 + */ + +@RestController +@RequestMapping("/chuanda/foreignExchange") +@Api(value = "API - 川大:外汇市场人民币品种", tags = "川大子系统:外汇市场") +public class WhscRenminbipinzhongController { + + @Resource + public WhscRenminbipinzhongService service; + + /** + * @描述: 列表:根据时间、币种条件查询 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-09-28 + **/ + @ApiOperation(value = "当日外汇行情:根据时间、币种条件查询", response = WhscRenminbipinzhongEntity.class) + @PostMapping(value = "/pageCondition") + public R pageCondition(@RequestBody CdPageConditionReq entity) { + + Integer userId = TokenUtil.getUserId(); + //TODO 日期格式化 + String dateFormat = entity.getDate(); + PageUtils list = service.queryPageByRecode(entity.getPageNum(), entity.getPageSize(), entity.getName(), dateFormat, userId); + return R.ok().put("data", list); + } + + + /** + * @描述: 查看外汇详情 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-09-28 + **/ + @ApiOperation(value = "查看外汇详情", response = WhscRenminbipinzhongEntity.class) + @PostMapping(value = "/detail") + public R detail(@RequestParam("id") @ApiParam(value = "序号") Integer id) { + Integer userId = TokenUtil.getUserId(); + WhscRenminbipinzhongEntity entity = service.getById(id); + return R.ok().put("data", entity); + } + + + /** + * @描述: 查看外汇每周/每月的信息 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-09-28 + **/ + @ApiOperation(value = "查看外汇每周/每月的信息", response = CdGetWeekAndMonthInfoResp.class) + @PostMapping(value = "/getWeekAndMonthInfo") + public Map getWeekAndMonthInfo(@RequestBody CdGetWeekAndMonthInfoReq req) { + + Map map = new HashMap(); + + Integer userId = TokenUtil.getUserId(); + + + //1.根据id查询到该条外汇信息 + WhscRenminbipinzhongEntity entity = service.getById(req.getId()); + + //2.根据外汇名称查询该合约一周的信息 + QueryWrapper wrapper = new QueryWrapper<>(); + wrapper.eq("name", entity.getName()); + wrapper.last(" and YEARWEEK(date_format(date,'%Y-%m-%d')) = YEARWEEK('" + req.getDate() + "') ORDER BY date "); + List week = service.list(wrapper); + + /*List getWeek = new ArrayList(); + + for (WhscRenminbipinzhongEntity list : week) { + CdGetWeekAndMonthInfoResp resp = new CdGetWeekAndMonthInfoResp(); + resp.setDate(list.getDate()); + resp.setLastPrice(list.getLastPrice()); + getWeek.add(resp); + + }*/ + + + //3.根据外汇名称查询该合约一个月的信息 + + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("name", entity.getName()); + queryWrapper.last(" AND DATE_FORMAT( date, '%Y%m' ) = DATE_FORMAT( '" + req.getDate() + "' , '%Y%m' ) ORDER BY date "); + List month = service.list(queryWrapper); + + /* List getMonth = new ArrayList<>(); + + for (WhscRenminbipinzhongEntity list1 : month) { + CdGetWeekAndMonthInfoResp resp1 = new CdGetWeekAndMonthInfoResp(); + resp1.setDate(list1.getDate()); + resp1.setLastPrice(list1.getLastPrice()); + getMonth.add(resp1); + + }*/ + + map.put("errmessage", "success"); + map.put("status", 200); + map.put("week", week); + map.put("month", month); + return map; + + } + + + /** + * 根据时间、币种条件查询后导出报表 + * + * @param exportReq + * @param response + * @return + */ + @ApiOperation(value = "根据时间、币种条件查询后导出报表", notes = "根据时间、币种条件查询后导出报表") + @RequestMapping(value = "/export", method = RequestMethod.GET, produces = "application/json") + public R export(CdExportReq exportReq, HttpServletResponse response) { + try { + //TODO:历史数据下载:只能选取所选择的行情日期之前的数据下载 + R r = service.exportProjectRecord(response, exportReq.getName(), exportReq.getStartTime(), exportReq.getEndTime()); + return r; + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + + } + + + /** + * @描述: 根据选择的模拟法传值获取相应数据 + * @入参: + * @出参: + * @作者: Rong + * @日期: 2020-09-28 + **/ + @ApiOperation(value = "根据选择的模拟法传值获取相应数据", response = CdCreatePhotoVo.class) + @PostMapping(value = "/simulation", produces = MediaType.APPLICATION_JSON_UTF8_VALUE) + @CrossOrigin + public R simulation(@RequestBody CdSimulationReq req) { + + try { + Integer userId = TokenUtil.getUserId(); + //TODO:历史数据下载:只能选取所选择的行情日期之前的数据下载 + R r = service.getInfobyId(req.getName(), req.getStartTime(), req.getEndTime(), req.getConfidenceLevel(), req.getValidity(), req.getType(), req.getExperimentsNumber()); + + return r; + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + + + } + + public static void main(String[] args) { + //连接本地的 Redis 服务 + Jedis jedis = new Jedis("localhost"); + System.out.println("连接成功"); + //查看服务是否运行 + System.out.println("服务正在运行: " + jedis.ping()); + } + + +} diff --git a/src/main/java/com/msdw/tms/dao/CdBankStatementsDao.java b/src/main/java/com/msdw/tms/dao/CdBankStatementsDao.java new file mode 100644 index 0000000..9945795 --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/CdBankStatementsDao.java @@ -0,0 +1,16 @@ +package com.msdw.tms.dao; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.msdw.tms.entity.CdBankStatementsEntity; +import org.apache.ibatis.annotations.Mapper; + +/** + * @描述:账户流水记录 Mapper 接口 + * @作者: Rong + * @日期: 2020-10-15 + */ +@Mapper +public interface CdBankStatementsDao extends BaseMapper { + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/CdCacheDao.java b/src/main/java/com/msdw/tms/dao/CdCacheDao.java new file mode 100644 index 0000000..d88af8c --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/CdCacheDao.java @@ -0,0 +1,15 @@ +package com.msdw.tms.dao; + +import org.apache.ibatis.annotations.Mapper; +import com.msdw.tms.entity.CdCacheEntity; +import com.baomidou.mybatisplus.core.mapper.BaseMapper; + +/** + * @描述:步骤缓存记录表 Mapper 接口 + * @作者: Rong + * @日期: 2020-10-21 + */ +@Mapper +public interface CdCacheDao extends BaseMapper { + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/CdExperimentalDao.java b/src/main/java/com/msdw/tms/dao/CdExperimentalDao.java new file mode 100644 index 0000000..0d8721b --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/CdExperimentalDao.java @@ -0,0 +1,15 @@ +package com.msdw.tms.dao; + +import org.apache.ibatis.annotations.Mapper; +import com.msdw.tms.entity.CdExperimentalEntity; +import com.baomidou.mybatisplus.core.mapper.BaseMapper; + +/** + * @描述:实验数据 Mapper 接口 + * @作者: Rong + * @日期: 2020-10-20 + */ +@Mapper +public interface CdExperimentalDao extends BaseMapper { + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/CdFuturesConfigureDao.java b/src/main/java/com/msdw/tms/dao/CdFuturesConfigureDao.java new file mode 100644 index 0000000..ae109fb --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/CdFuturesConfigureDao.java @@ -0,0 +1,15 @@ +package com.msdw.tms.dao; + +import org.apache.ibatis.annotations.Mapper; +import com.msdw.tms.entity.CdFuturesConfigureEntity; +import com.baomidou.mybatisplus.core.mapper.BaseMapper; + +/** + * @描述:期货资产头寸配置 Mapper 接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Mapper +public interface CdFuturesConfigureDao extends BaseMapper { + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/CdPointChildrenDao.java b/src/main/java/com/msdw/tms/dao/CdPointChildrenDao.java new file mode 100644 index 0000000..bb9f8ea --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/CdPointChildrenDao.java @@ -0,0 +1,15 @@ +package com.msdw.tms.dao; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.msdw.tms.entity.CdPointChildren; +import org.apache.ibatis.annotations.Mapper; + +/** + * @描述:判分点值子级表 Mapper 接口 + * @作者: Rong + * @日期: 2021-01-12 + */ +@Mapper +public interface CdPointChildrenDao extends BaseMapper { + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/CdPointDao.java b/src/main/java/com/msdw/tms/dao/CdPointDao.java new file mode 100644 index 0000000..b1a64ab --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/CdPointDao.java @@ -0,0 +1,15 @@ +package com.msdw.tms.dao; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.msdw.tms.entity.CdPoint; +import org.apache.ibatis.annotations.Mapper; + +/** + * @描述: Mapper 接口 + * @作者: Rong + * @日期: 2021-01-11 + */ +@Mapper +public interface CdPointDao extends BaseMapper { + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/CdTransferWarehouseRecordDao.java b/src/main/java/com/msdw/tms/dao/CdTransferWarehouseRecordDao.java new file mode 100644 index 0000000..35943e7 --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/CdTransferWarehouseRecordDao.java @@ -0,0 +1,15 @@ +package com.msdw.tms.dao; + +import org.apache.ibatis.annotations.Mapper; +import com.msdw.tms.entity.CdTransferWarehouseRecordEntity; +import com.baomidou.mybatisplus.core.mapper.BaseMapper; + +/** + * @描述:调仓记录表 Mapper 接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Mapper +public interface CdTransferWarehouseRecordDao extends BaseMapper { + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/CdUserAssetsDao.java b/src/main/java/com/msdw/tms/dao/CdUserAssetsDao.java new file mode 100644 index 0000000..b573f79 --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/CdUserAssetsDao.java @@ -0,0 +1,15 @@ +package com.msdw.tms.dao; + +import org.apache.ibatis.annotations.Mapper; +import com.msdw.tms.entity.CdUserAssetsEntity; +import com.baomidou.mybatisplus.core.mapper.BaseMapper; + +/** + * @描述:用户银行账户 Mapper 接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Mapper +public interface CdUserAssetsDao extends BaseMapper { + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/CdUserOptionAccountDao.java b/src/main/java/com/msdw/tms/dao/CdUserOptionAccountDao.java new file mode 100644 index 0000000..2d5de64 --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/CdUserOptionAccountDao.java @@ -0,0 +1,15 @@ +package com.msdw.tms.dao; + +import org.apache.ibatis.annotations.Mapper; +import com.msdw.tms.entity.CdUserOptionAccountEntity; +import com.baomidou.mybatisplus.core.mapper.BaseMapper; + +/** + * @描述:期货期权账户 Mapper 接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Mapper +public interface CdUserOptionAccountDao extends BaseMapper { + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/QhscGangjiaosuoDao.java b/src/main/java/com/msdw/tms/dao/QhscGangjiaosuoDao.java new file mode 100644 index 0000000..0b0a30c --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/QhscGangjiaosuoDao.java @@ -0,0 +1,28 @@ +package com.msdw.tms.dao; + +import com.baomidou.dynamic.datasource.annotation.DS; +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.msdw.tms.entity.QhscGangjiaosuoEntity; +import org.apache.ibatis.annotations.Mapper; +import org.apache.ibatis.annotations.Param; + +import java.util.List; + +/** + * @描述: Mapper 接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Mapper +@DS("pydb") +public interface QhscGangjiaosuoDao extends BaseMapper { + IPage queryPageByQh(Page page, @Param("name") String name, @Param("date") String date); + + + //查询所有币种 + List getName(); + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/dao/WhscRenminbipinzhongDao.java b/src/main/java/com/msdw/tms/dao/WhscRenminbipinzhongDao.java new file mode 100644 index 0000000..493b96d --- /dev/null +++ b/src/main/java/com/msdw/tms/dao/WhscRenminbipinzhongDao.java @@ -0,0 +1,25 @@ +package com.msdw.tms.dao; + +import com.baomidou.dynamic.datasource.annotation.DS; +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.msdw.tms.entity.WhscRenminbipinzhongEntity; +import org.apache.ibatis.annotations.Mapper; + +import java.util.List; + +/** + * @描述:外汇市场人民币品种 Mapper 接口 + * @作者: Rong + * @日期: 2020-09-28 + */ + +@Mapper +@DS("pydb") +public interface WhscRenminbipinzhongDao extends BaseMapper { + + /*汇率风险识别历史行情数据下载 根据时间区间查询*/ + List exportByDate(String startTime,String endTime); + + //查询所有汇率类型 + List getExchangeRate(); +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/CdBankStatementsEntity.java b/src/main/java/com/msdw/tms/entity/CdBankStatementsEntity.java new file mode 100644 index 0000000..c057215 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/CdBankStatementsEntity.java @@ -0,0 +1,57 @@ +package com.msdw.tms.entity; + +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotBlank; +import java.io.Serializable; +import java.math.BigDecimal; + +/** + * @描述:账户流水记录 + * @作者: Rong + * @日期: 2020-10-15 + */ +@Data +@ApiModel(value = "账户流水记录") +@TableName("cd_bank_statements") +public class CdBankStatementsEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + private Long id; + @ApiModelProperty(value = "用户id") + @NotBlank(message = "用户id不能为空") + private Integer userId; + + @ApiModelProperty(value = "类型(1银行账户 2期货账户)") + private Integer type; + + @ApiModelProperty(value = "币种") + private String coinType; + + @ApiModelProperty(value = "改变前金额") + private BigDecimal beforeAmount; + + @ApiModelProperty(value = "金额") + private BigDecimal amountMoney; + + @ApiModelProperty(value = "改变后金额") + private BigDecimal afterAmount; + + @ApiModelProperty(value = "流水类型(1:转账 2:流水)") + private Integer transferType; + + @ApiModelProperty(value = "备注") + private String remark; + + @ApiModelProperty(value = "创建时间") + private String createTime; + + @ApiModelProperty(value = "修改时间") + private String updateTime; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/CdCacheEntity.java b/src/main/java/com/msdw/tms/entity/CdCacheEntity.java new file mode 100644 index 0000000..1caa641 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/CdCacheEntity.java @@ -0,0 +1,62 @@ +package com.msdw.tms.entity; + +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.io.Serializable; + +/** + * @描述:步骤缓存记录表 + * @作者: Rong + * @日期: 2020-10-21 + */ +@Data +@ApiModel(value = "步骤缓存记录表") +@TableName("cd_cache") +public class CdCacheEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + @ApiModelProperty(value = "主键") + private Integer id; + + @ApiModelProperty(value = "用户id") + private Integer userId; + + @ApiModelProperty(value = "步骤标识(标识为用户做到哪一步骤了)") + private Integer stepNumber; + + @ApiModelProperty(value = "步骤1 Json字符串") + private String step1; + + @ApiModelProperty(value = "步骤2 Json字符串") + private String step2; + + @ApiModelProperty(value = "步骤3 Json字符串") + private String step3; + + @ApiModelProperty(value = "步骤4 Json字符串") + private String step4; + + @ApiModelProperty(value = "步骤5 Json字符串") + private String step5; + + @ApiModelProperty(value = "步骤6 Json字符串") + private String step6; + + @ApiModelProperty(value = "步骤7 Json字符串") + private String step7; + + @ApiModelProperty(value = "步骤8 Json字符串") + private String step8; + + @ApiModelProperty(value = "步骤9 Json字符串") + private String step9; + + @ApiModelProperty(value = "步骤10 Json字符串") + private String step10; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/CdExperimentalEntity.java b/src/main/java/com/msdw/tms/entity/CdExperimentalEntity.java new file mode 100644 index 0000000..6d278f8 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/CdExperimentalEntity.java @@ -0,0 +1,74 @@ +package com.msdw.tms.entity; + +import cn.afterturn.easypoi.excel.annotation.Excel; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.io.Serializable; + +/** + * @描述:实验数据 + * @作者: Rong + * @日期: 2020-10-20 + */ +@Data +@ApiModel(value = "实验数据") +@TableName("cd_experimental") +public class CdExperimentalEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + @Excel(name = "序号", orderNum = "1", mergeVertical = true, isImportField = "id", width = 15) + @ApiModelProperty(value = "主键") + private Integer id; + + @ApiModelProperty(value = "用户id") + private Integer userId; + + @ApiModelProperty(value = "实验类型(0正态Var,1历史模拟,2蒙特卡洛)") + private Integer type; + + + @Excel(name = "汇率类型", orderNum = "2", mergeVertical = true, isImportField = "exchangeRate", width = 15) + @ApiModelProperty(value = "汇率类型") + private String exchangeRate; + + + @Excel(name = "样本数据", orderNum = "3", mergeVertical = true, isImportField = "sampleData", width = 25) + @ApiModelProperty(value = "样本数据") + private String sampleData; + + + @Excel(name = "置信水平", orderNum = "4", mergeVertical = true, isImportField = "confidenceLevel", width = 15) + @ApiModelProperty(value = "置信水平") + private String confidenceLevel; + + @Excel(name = "持有天数", orderNum = "5", mergeVertical = true, isImportField = "validPeriod", width = 15) + @ApiModelProperty(value = "持有天数") + private String validPeriod; + + @ApiModelProperty(value = "实验次数") + private Integer experimentsNumber; + + + @Excel(name = "Var值", orderNum = "6", mergeVertical = true, isImportField = "varCode", width = 20) + @ApiModelProperty(value = "Var值") + private String varCode; + + @ApiModelProperty(value = "汇率走势图") + private String trendChart; + + @ApiModelProperty(value = "汇率损益分布图") + private String profitLossDistribution; + + @Excel(name = "创建时间", orderNum = "7", mergeVertical = true, isImportField = "creatTime", width = 20) + @ApiModelProperty(value = "创建时间") + private String creatTime; + + @ApiModelProperty(value = "修改时间") + private String updateTime; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/CdFuturesConfigureEntity.java b/src/main/java/com/msdw/tms/entity/CdFuturesConfigureEntity.java new file mode 100644 index 0000000..f3f558b --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/CdFuturesConfigureEntity.java @@ -0,0 +1,148 @@ +package com.msdw.tms.entity; + +import com.baomidou.mybatisplus.annotation.IdType; +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 javax.validation.constraints.NotBlank; +import javax.validation.constraints.NotNull; +import java.io.Serializable; +import java.math.BigDecimal; + +/** + * @描述:建仓记录表 + * @作者: Rong + * @日期: 2020-10-12 + */ +@Data +@ApiModel(value = "建仓记录表") +@TableName("cd_futures_configure") +public class CdFuturesConfigureEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + @ApiModelProperty(value = "主键", required = false) + @TableId(type = IdType.AUTO) + private Integer id; + + @ApiModelProperty(value = "用户id") + @NotNull(message = "用户id不能为空") + private Integer userId; + + @ApiModelProperty(value = "合约表主键") + @NotNull(message = "合约表主键不能为空") + private Integer pid; + + @ApiModelProperty(value = "交易方向(0空头,1多头)") + @NotBlank(message = "交易方向不能为空") + private Integer tradingDirection; + + @ApiModelProperty(value = "最新价格") + @NotBlank(message = "最新价格不能为空") + private BigDecimal newestPrice; + + @ApiModelProperty(value = "委托价格") + @NotNull(message = "委托价格不能为空") + private BigDecimal entrustPrice; + + @ApiModelProperty(value = "委托数量") + @NotNull(message = "委托数量不能为空") + private Integer entrustNumber; + + @ApiModelProperty(value = "可交易数量") + @NotNull(message = "可交易数量不能为空") + private BigDecimal tradableNumber; + + @ApiModelProperty(value = "预冻结保证金") + private BigDecimal preFreezingMargin; + + @ApiModelProperty(value = "预冻结手续费") + private BigDecimal preFreezingFrozen; + + @ApiModelProperty(value = "所需资金") + private BigDecimal fundingNeeded; + + @ApiModelProperty(value = "冻结资金") + private BigDecimal frozenPrice; + + @ApiModelProperty(value = "0.开仓 1.平仓") + private Integer status; + + @ApiModelProperty(value = "是否自动平仓(0否 1是)") + private Integer type; + + @ApiModelProperty(value = "浮动盈亏") + private BigDecimal floatingPl; + + @ApiModelProperty(value = "开仓时间") + @NotBlank(message = "开仓时间不能为空") + private String createTime; + + @ApiModelProperty(value = "平仓时间") + @NotBlank(message = "平仓时间不能为空") + private String updateTime; + + @ApiModelProperty(value = "最后交易日") + @NotBlank(message = "最后交易日不能为空") + private String tradeTime; + + + @ApiModelProperty(value = "平仓价格") + @NotBlank(message = "平仓价格不能为空") + private BigDecimal closeRate; + + //transient设置非数据库字段 + /** + * 浮动盈亏 + */ + //private transient String floatingPL; + + /** + * 名称 + */ + @ApiModelProperty(value = "名称") + private transient String name; + + /** + * 代码 + */ + @ApiModelProperty(value = "代码") + private transient String ename; + + + /** + * 当前最新价 + */ + @ApiModelProperty(value = "当前最新价") + private transient BigDecimal lastPrice; + + /** + * 涨跌幅 + */ + @ApiModelProperty(value = "涨跌幅") + private transient String u; + + + /** + * 昨收 + */ + @ApiModelProperty(value = "昨结") + private transient BigDecimal dayUp; + + //买卖方向为开仓时候:可交易数量为Python表的持仓总量 + @ApiModelProperty(value = "买卖方向为开仓时候:可交易数量为Python表的持仓总量") + private transient BigDecimal tradableQuantityByBuy; + + //买卖方向为平仓时候:可交易数量为自己的持仓总量 + @ApiModelProperty(value = "买卖方向为平仓时候:可交易数量为自己的持仓总量") + private transient BigDecimal tradableQuantityBySell; + + //当前账户期权可用资金 + @ApiModelProperty(value = "当前期权账户可用资金") + private transient BigDecimal availableFunds; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/CdPoint.java b/src/main/java/com/msdw/tms/entity/CdPoint.java new file mode 100644 index 0000000..ea06269 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/CdPoint.java @@ -0,0 +1,35 @@ +package com.msdw.tms.entity; + +import com.baomidou.mybatisplus.annotation.IdType; +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 java.io.Serializable; + +/** + * @描述: + * @作者: Rong + * @日期: 2021-01-11 + */ +@Data +@ApiModel(value = "判分点值设置") +@TableName("cd_point") +public class CdPoint implements Serializable { + + private static final long serialVersionUID = 1L; + + @TableId(type = IdType.AUTO) + @ApiModelProperty(value = "主键") + private Integer id; + + @ApiModelProperty(value = "名称") + private String name; + + @ApiModelProperty(value = "父级id") + private Integer parentId; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/CdPointChildren.java b/src/main/java/com/msdw/tms/entity/CdPointChildren.java new file mode 100644 index 0000000..116fc46 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/CdPointChildren.java @@ -0,0 +1,39 @@ +package com.msdw.tms.entity; + +import com.baomidou.mybatisplus.annotation.IdType; +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 java.io.Serializable; + +/** + * @描述:判分点值子级表 + * @作者: Rong + * @日期: 2021-01-12 + */ +@Data +@ApiModel(value = "判分点值子级表") +@TableName("cd_point_children") +public class CdPointChildren implements Serializable { + + private static final long serialVersionUID = 1L; + + @TableId(type = IdType.AUTO) + @ApiModelProperty(value = "主键") + private Integer cid; + + @ApiModelProperty(value = "名称") + private String childrenName; + + @ApiModelProperty(value = "父级id") + private Integer pointId; + + + @ApiModelProperty(value = "类型(1.选择题 2.填空题 3.问答题 4.指标结果 )") + private String type; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/CdTransferWarehouseRecordEntity.java b/src/main/java/com/msdw/tms/entity/CdTransferWarehouseRecordEntity.java new file mode 100644 index 0000000..dd2f2bc --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/CdTransferWarehouseRecordEntity.java @@ -0,0 +1,52 @@ +package com.msdw.tms.entity; + +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotNull; +import java.io.Serializable; +import java.math.BigDecimal; + +/** + * @描述:调仓记录表 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Data +@ApiModel(value = "操作记录表") +@TableName("cd_transfer_warehouse_record") +public class CdTransferWarehouseRecordEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + private Integer id; + + @ApiModelProperty(value = "头寸主键") + private Integer cid; + + @ApiModelProperty(value = "用户id") + @NotNull(message = "用户id不能为空") + private Integer userId; + + @ApiModelProperty(value = "金额") + private BigDecimal balance; + + @ApiModelProperty(value = "买卖方向(0开仓 1平仓)") + private Integer direction; + + @ApiModelProperty(value = "委托类型(0市价 1限价)") + private Integer type; + + @ApiModelProperty(value = "委托手数") + private BigDecimal number; + + @ApiModelProperty(value = "创建时间") + private String createTime; + + @ApiModelProperty(value = "修改时间") + private String updateTime; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/CdUserAssetsEntity.java b/src/main/java/com/msdw/tms/entity/CdUserAssetsEntity.java new file mode 100644 index 0000000..c90aa04 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/CdUserAssetsEntity.java @@ -0,0 +1,50 @@ +package com.msdw.tms.entity; + +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.io.Serializable; +import java.math.BigDecimal; + +/** + * @描述:用户银行账户 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Data +@ApiModel(value = "用户银行账户") +@TableName("cd_user_assets") +public class CdUserAssetsEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + private Long id; + + @ApiModelProperty(value = "用户id") + private Long userId; + + @ApiModelProperty(value = "初始资金") + private BigDecimal initialBalance; + + @ApiModelProperty(value = "可用资金") + private BigDecimal balance; + + @ApiModelProperty(value = "创建时间") + private String createTime; + + @ApiModelProperty(value = "修改时间") + private String updateTime; + + + public CdUserAssetsEntity() { + } + + public CdUserAssetsEntity(Long userId, BigDecimal initialBalance, BigDecimal balance, String createTime ) { + this.userId = userId; + this.initialBalance = initialBalance; + this.balance = balance; + this.createTime = createTime; + } +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/CdUserOptionAccountEntity.java b/src/main/java/com/msdw/tms/entity/CdUserOptionAccountEntity.java new file mode 100644 index 0000000..d1d31b5 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/CdUserOptionAccountEntity.java @@ -0,0 +1,56 @@ +package com.msdw.tms.entity; + +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.io.Serializable; +import java.math.BigDecimal; + +/** + * @描述:期货期权账户 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Data +@ApiModel(value = "期货期权账户") +@TableName("cd_user_option_account") +public class CdUserOptionAccountEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + private Long id; + + @ApiModelProperty(value = "用户id") + private Long userId; + + @ApiModelProperty(value = "可用资金") + private BigDecimal balance; + + @ApiModelProperty(value = "冻结资金") + private BigDecimal forzenBalance; + + @ApiModelProperty(value = "占用保证金") + private BigDecimal depositBalance; + + @ApiModelProperty(value = "创建时间") + private String createTime; + + @ApiModelProperty(value = "修改时间") + private String updateTime; + + @ApiModelProperty(value = "总资产") + private transient BigDecimal totalAssets; + + public CdUserOptionAccountEntity() { + } + + public CdUserOptionAccountEntity(Long userId, BigDecimal balance, BigDecimal forzenBalance, BigDecimal depositBalance, String createTime) { + this.userId = userId; + this.balance = balance; + this.forzenBalance = forzenBalance; + this.depositBalance = depositBalance; + this.createTime = createTime; + } +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/QhscGangjiaosuoEntity.java b/src/main/java/com/msdw/tms/entity/QhscGangjiaosuoEntity.java new file mode 100644 index 0000000..d6f9f80 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/QhscGangjiaosuoEntity.java @@ -0,0 +1,79 @@ +package com.msdw.tms.entity; + +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotNull; +import java.io.Serializable; +import java.math.BigDecimal; + +/** + * @描述: + * @作者: Rong + * @日期: 2020-10-10 + */ +@Data +@ApiModel(value = "") +@TableName("qhsc_gangjiaosuo") +public class QhscGangjiaosuoEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + private Integer id; + + @ApiModelProperty(value = "代码") + @NotNull + private String eName; + + @ApiModelProperty(value = "名称") + private String name; + + @ApiModelProperty(value = "最新价") + private BigDecimal lastPrice; + + @ApiModelProperty(value = "涨跌幅") + private String u; + + @ApiModelProperty(value = "涨跌额") + private BigDecimal udPrice; + + @ApiModelProperty(value = "买入价") + private BigDecimal buyPrice; + + @ApiModelProperty(value = "卖出价") + private BigDecimal cellPrice; + + @ApiModelProperty(value = "买量") + private Integer buy; + + @ApiModelProperty(value = "卖量") + private Integer cell; + + @ApiModelProperty(value = "总量") + private BigDecimal suoyou; + + @ApiModelProperty(value = "现量") + private BigDecimal nowHava; + + @ApiModelProperty(value = "持仓量") + private String hava; + + @ApiModelProperty(value = "日增") + private Double yFinish; + + @ApiModelProperty(value = "昨结") + private BigDecimal dayUp; + + @ApiModelProperty(value = "日期") + private String date; + + + /*非数据库字段*/ + @ApiModelProperty(value = "币种") + private transient String currency; + + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/WhscRenminbipinzhongEntity.java b/src/main/java/com/msdw/tms/entity/WhscRenminbipinzhongEntity.java new file mode 100644 index 0000000..0ff36f3 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/WhscRenminbipinzhongEntity.java @@ -0,0 +1,190 @@ +package com.msdw.tms.entity; + +import cn.afterturn.easypoi.excel.annotation.Excel; +import cn.afterturn.easypoi.excel.annotation.ExcelTarget; +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 java.io.Serializable; + +/** + * @描述:外汇市场人民币品种 + * @作者: Rong + * @日期: 2020-09-28 + */ + +//@EqualsAndHashCode(callSuper = true) +@ExcelTarget("whsc_renminbipinzhong") +@TableName("whsc_renminbipinzhong") +@ApiModel(value = "外汇市场人民币品种") + +public class WhscRenminbipinzhongEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + + @TableId("id") + @Excel(name = "编号", orderNum = "1", mergeVertical = true, isImportField = "id", width = 15) + private Integer id; + + + @TableField("e_name") + @Excel(name = "代码", orderNum = "2", mergeVertical = true, isImportField = "eName", width = 15) + @ApiModelProperty(value = "代码") + private String eName; + + @TableField("name") + @Excel(name = "名称", orderNum = "3", mergeVertical = true, isImportField = "name", width = 30) + @ApiModelProperty(value = "名称") + private String name; + + @TableField("last_price") + @Excel(name = "最新价", orderNum = "4", mergeVertical = true, isImportField = "lastPrice", width = 15) + @ApiModelProperty(value = "最新价") + private Double lastPrice; + + + @TableField("ud_price") + @Excel(name = "涨跌额", orderNum = "5", mergeVertical = true, isImportField = "udPrice", width = 15) + @ApiModelProperty(value = "涨跌额") + private Double udPrice; + + + @TableField("u") + @Excel(name = "涨跌幅", orderNum = "6", mergeVertical = true, isImportField = "u", width = 15) + @ApiModelProperty(value = "涨跌幅") + private String u; + + + @TableField("t_open") + @Excel(name = "今开", orderNum = "15", mergeVertical = true, isImportField = "tOpen", width = 15) + @ApiModelProperty(value = "今开") + private Double tOpen; + + + @TableField("high") + @Excel(name = "最高", orderNum = "8", mergeVertical = true, isImportField = "high", width = 15) + @ApiModelProperty(value = "最高") + private Double high; + + + @TableField("low") + @Excel(name = "最低", orderNum = "9", mergeVertical = true, isImportField = "low", width = 15) + @ApiModelProperty(value = "最低") + private Double low; + + @TableField("y_finish") + @Excel(name = "昨结", orderNum = "10", mergeVertical = true, isImportField = "yFinish", width = 15) + @ApiModelProperty(value = "昨结") + private Double yFinish; + + + @TableField("date") + @Excel(name = "时间", orderNum = "11", mergeVertical = true, isImportField = "date", width = 25) + @ApiModelProperty(value = "时间") + private String date; + + + @ApiModelProperty(value = "汇率类型") + private transient String typeName; + + + public Integer getId() { + return id; + } + + public void setId(Integer id) { + this.id = id; + } + + public String geteName() { + return eName; + } + + public void seteName(String eName) { + this.eName = eName; + } + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } + + public Double getLastPrice() { + return lastPrice; + } + + public void setLastPrice(Double lastPrice) { + this.lastPrice = lastPrice; + } + + public Double getUdPrice() { + return udPrice; + } + + public void setUdPrice(Double udPrice) { + this.udPrice = udPrice; + } + + public String getU() { + return u; + } + + public void setU(String u) { + this.u = u; + } + + public Double gettOpen() { + return tOpen; + } + + public void settOpen(Double tOpen) { + this.tOpen = tOpen; + } + + public Double getHigh() { + return high; + } + + public void setHigh(Double high) { + this.high = high; + } + + public Double getLow() { + return low; + } + + public void setLow(Double low) { + this.low = low; + } + + public Double getyFinish() { + return yFinish; + } + + public void setyFinish(Double yFinish) { + this.yFinish = yFinish; + } + + public String getDate() { + return date; + } + + public void setDate(String date) { + this.date = date; + } + + public String getTypeName() { + return typeName; + } + + public void setTypeName(String typeName) { + this.typeName = typeName; + } +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/request/ProjectRecordImportRequest.java b/src/main/java/com/msdw/tms/entity/request/ProjectRecordImportRequest.java new file mode 100644 index 0000000..b250786 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/request/ProjectRecordImportRequest.java @@ -0,0 +1,41 @@ +package com.msdw.tms.entity.request; + +import com.msdw.tms.common.utils.poi.ExcelAttribute; +import lombok.Data; + +@Data +public class ProjectRecordImportRequest { + + /** + * 实验项目名称 + */ + @ExcelAttribute(sort = 0) + private String projectName; + /** + * 状态 + */ + @ExcelAttribute(sort = 1) + private String userRecordstate; + + private Integer recordstate; + /** + * 得分 + */ + @ExcelAttribute(sort = 2) + private Integer score; + /** + * 耗时 + */ + @ExcelAttribute(sort = 3) + private Integer timeSum; + /** + * 起始时间 + */ + @ExcelAttribute(sort = 4) + private String startTime; + /** + * 结束时间 + */ + @ExcelAttribute(sort = 5) + private String endTime; +} diff --git a/src/main/java/com/msdw/tms/entity/request/QuestionsAddRequest.java b/src/main/java/com/msdw/tms/entity/request/QuestionsAddRequest.java new file mode 100644 index 0000000..5da4a57 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/request/QuestionsAddRequest.java @@ -0,0 +1,67 @@ +package com.msdw.tms.entity.request; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotNull; + +/** + * 试题的基本信息表 + * + * @author gongsj + */ +@Data +@ApiModel(value = "QuestionsAddRequest", description = "添加试题请求体对象") +public class QuestionsAddRequest { + /** + * 题型:1、单选题,2、多选题,3、判断题 + */ + @ApiModelProperty(value = "题型:1、单选题,2、多选题,3、判断题", name = "questionType", example = "1", required = true) + private Integer questionType; + /** + * 题干信息 + */ + @ApiModelProperty(value = "题干信息", name = "questionStem", example = "世界上最大的哺乳动物是什么?", required = true) + private String questionStem; + /** + * A选项内容 + */ + @ApiModelProperty(value = "A选项内容", name = "optionA", example = "鲸鱼") + private String optionA; + /** + * B选项内容 + */ + @ApiModelProperty(value = "B选项内容", name = "optionB", example = "鲨鱼") + private String optionB; + /** + * C选项内容 + */ + @ApiModelProperty(value = "C选项内容", name = "optionC", example = "河马") + private String optionC; + /** + * D选项内容 + */ + @ApiModelProperty(value = "D选项内容", name = "optionD", example = "豹子") + private String optionD; + /** + * E选项内容 + */ + @ApiModelProperty(value = "E选项内容", name = "optionE", example = "大象") + private String optionE; + /** + * F选项内容 + */ + @ApiModelProperty(value = "F选项内容", name = "optionF", example = "老虎") + private String optionF; + /** + * 正确答案 + */ + @ApiModelProperty(value = "正确答案", name = "answer", example = "A", required = true) + private String answer; + /** + * 答案解析 + */ + @ApiModelProperty(value = "答案解析", name = "answerAnalysis", example = "鲸鱼是最大的哺乳动物") + private String answerAnalysis; +} diff --git a/src/main/java/com/msdw/tms/entity/request/QuestionsImportRequest.java b/src/main/java/com/msdw/tms/entity/request/QuestionsImportRequest.java new file mode 100644 index 0000000..d9435ff --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/request/QuestionsImportRequest.java @@ -0,0 +1,81 @@ +package com.msdw.tms.entity.request; + +import com.msdw.tms.common.utils.poi.ExcelAttribute; +import lombok.Data; +import lombok.ToString; + +/** + * 试题的基本信息表 + * + * @author gongsj + */ +@Data +@ToString +public class QuestionsImportRequest { + /** + * 第几行 + */ + private Integer index; + + /** + * 题干,问题描述 + */ + @ExcelAttribute(sort = 0) + private String questionStem; + /** + * 题型 + */ + @ExcelAttribute(sort = 1) + private String questionTypeName; + /** + * 选项A + */ + @ExcelAttribute(sort = 2) + private String optionA; + /** + * 选项B + */ + @ExcelAttribute(sort = 3) + private String optionB; + /** + * 选项C + */ + @ExcelAttribute(sort = 4) + private String optionC; + /** + * 选项D + */ + @ExcelAttribute(sort = 5) + private String optionD; + /** + * 选项E + */ + @ExcelAttribute(sort = 6) + private String optionE; + /** + * 选项F + */ + @ExcelAttribute(sort = 7) + private String optionF; + /** + * 正确答案 + */ + @ExcelAttribute(sort = 8) + private String answer; + /** + * 答案解析 + */ + @ExcelAttribute(sort = 9) + private String answerAnalysis; + + public String toStringForCompare() { + return questionStem + + questionTypeName + + optionA + + optionB + + optionC + + optionD + + optionE + + optionF; + } +} diff --git a/src/main/java/com/msdw/tms/entity/request/QuestionsQueryRequest.java b/src/main/java/com/msdw/tms/entity/request/QuestionsQueryRequest.java new file mode 100644 index 0000000..0e96b33 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/request/QuestionsQueryRequest.java @@ -0,0 +1,39 @@ +package com.msdw.tms.entity.request; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * 试题的基本信息表 + * + * @author gongsj + */ +@Data +@ApiModel(value = "QuestionsQueryRequest", description = "条件查询请求对象") +public class QuestionsQueryRequest { + + /** + * 题干,问题描述 + */ + @ApiModelProperty(value = "题干,问题描述", name = "questionStem", example = "世界上最大的哺乳动物是什么?") + private String questionStem; + + /** + * 题型 + */ + @ApiModelProperty(value = "题型:1、单选题,2、多选题,3、判断题", name = "questionType", example = "1") + private Integer questionType; + // /** + // * 参考答案 + // */ + // private String answer; + // /** + // * 答案解析 + // */ + // private String answerAnalysis; + // /** + // * 创建时,修改时间即为创建时间,修改时间用于前端显示和排序 + // */ + // private Date modifyTime; +} diff --git a/src/main/java/com/msdw/tms/entity/request/QuestionsUpdateRequest.java b/src/main/java/com/msdw/tms/entity/request/QuestionsUpdateRequest.java new file mode 100644 index 0000000..f44f6fa --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/request/QuestionsUpdateRequest.java @@ -0,0 +1,70 @@ +package com.msdw.tms.entity.request; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * 试题的基本信息表 + * + * @author gongsj + */ +@Data +@ApiModel(value = "QuestionsUpdateRequest", description = "修改试题请求体对象") +public class QuestionsUpdateRequest { + /** + * 主键 + */ + @ApiModelProperty(value = "主键", name = "id", example = "1", required = true) + private Integer id; + /** + * 题型:1、单选题,2、多选题,3、判断题 + */ + @ApiModelProperty(value = "题型:1、单选题,2、多选题,3、判断题", name = "questionType", example = "1", required = true) + private Integer questionType; + /** + * 题干信息 + */ + @ApiModelProperty(value = "题干信息", name = "questionStem", example = "世界上最大的哺乳动物是什么?", required = true) + private String questionStem; + /** + * A选项内容 + */ + @ApiModelProperty(value = "A选项内容", name = "optionA", example = "鲸鱼") + private String optionA; + /** + * B选项内容 + */ + @ApiModelProperty(value = "B选项内容", name = "optionB", example = "鲨鱼") + private String optionB; + /** + * C选项内容 + */ + @ApiModelProperty(value = "C选项内容", name = "optionC", example = "河马") + private String optionC; + /** + * D选项内容 + */ + @ApiModelProperty(value = "D选项内容", name = "optionD", example = "豹子") + private String optionD; + /** + * E选项内容 + */ + @ApiModelProperty(value = "E选项内容", name = "optionE", example = "大象") + private String optionE; + /** + * F选项内容 + */ + @ApiModelProperty(value = "F选项内容", name = "optionF", example = "老虎") + private String optionF; + /** + * 正确答案 + */ + @ApiModelProperty(value = "正确答案", name = "answer", example = "A", required = true) + private String answer; + /** + * 答案解析 + */ + @ApiModelProperty(value = "答案解析", name = "answerAnalysis", example = "鲸鱼是最大的哺乳动物") + private String answerAnalysis; +} diff --git a/src/main/java/com/msdw/tms/entity/response/CommonCode.java b/src/main/java/com/msdw/tms/entity/response/CommonCode.java new file mode 100644 index 0000000..b7c7ff9 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/response/CommonCode.java @@ -0,0 +1,51 @@ +package com.msdw.tms.entity.response; + +import lombok.ToString; + +@ToString +public enum CommonCode implements ResultCode { + + /* SUCCESS(true, 10000, "操作成功!"), + UNAUTHENTICATED(false, 10001, "此操作需要登陆系统!"), + UNAUTHORISE(false, 10002, "权限不足,无权操作!"), + INVALID_PARAM(false, 10003, "非法参数!"), + QUESTION_NUM_INVALID(false, 10004, "测评题目数量设置超出范围!"), + QUESTION_EXISTS(false, 10005, "此题已存在!"), + QUESTIONTYPE_INVALID(false, 10006, "题型错误!"), + EXCEL_INVALID(false, 10007, "excel表内容错误!"), + EVALUATION_TIME_INVALID(false, 10008, "测评时间错误!"), + EXCEL_FILE_INVALID(false, 10009, "上传excel文件错误!"), + REPEAT_INEXCEL(false, 10010, "试题在excel表中重复!"), + EVALUATION_QUESTION_NUM_INVALID(false, 10011, "当前测评题数设置为0,请先设置测评题数!"), + FAIL(false, 11111, "操作失败!"), + SERVER_ERROR(false, 99999, "抱歉,系统繁忙,请稍后重试!"); + //操作是否成功 + boolean success; + //操作代码 + int code; + //提示信息 + String message; + + CommonCode(boolean success, int code, String message) { + this.success = success; + this.code = code; + this.message = message; + } + + @Override + public boolean success() { + return success; + } + + @Override + public int code() { + return code; + } + + @Override + public String message() { + return message; + }*/ + + +} diff --git a/src/main/java/com/msdw/tms/entity/response/Response.java b/src/main/java/com/msdw/tms/entity/response/Response.java new file mode 100644 index 0000000..35b841e --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/response/Response.java @@ -0,0 +1,9 @@ +package com.msdw.tms.entity.response; + +/** + * Created by admin on 2018/3/5. + */ +public interface Response { + boolean SUCCESS = true; + int SUCCESS_CODE = 10000; +} diff --git a/src/main/java/com/msdw/tms/entity/response/ResponseResult.java b/src/main/java/com/msdw/tms/entity/response/ResponseResult.java new file mode 100644 index 0000000..9f6e454 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/response/ResponseResult.java @@ -0,0 +1,35 @@ +package com.msdw.tms.entity.response; + +import lombok.Data; +import lombok.NoArgsConstructor; +import lombok.ToString; + +@Data +@ToString +@NoArgsConstructor +public class ResponseResult implements Response { + + //操作是否成功 + /*boolean success = SUCCESS; + + //操作代码 + int code = SUCCESS_CODE; + + //提示信息 + String message; + + public ResponseResult(ResultCode resultCode) { + this.success = resultCode.success(); + this.code = resultCode.code(); + this.message = resultCode.message(); + } + + public static ResponseResult SUCCESS() { + return new ResponseResult(CommonCode.SUCCESS); + } + + public static ResponseResult FAIL() { + return new ResponseResult(CommonCode.FAIL); + }*/ + +} diff --git a/src/main/java/com/msdw/tms/entity/response/ResultCode.java b/src/main/java/com/msdw/tms/entity/response/ResultCode.java new file mode 100644 index 0000000..0bce5ff --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/response/ResultCode.java @@ -0,0 +1,21 @@ +package com.msdw.tms.entity.response; + +/** + * Created by mrt on 2018/3/5. + * 10000-- 通用错误代码 + * 22000-- 媒资错误代码 + * 23000-- 用户中心错误代码 + * 24000-- cms错误代码 + * 25000-- 文件系统 + */ +public interface ResultCode { + /* //操作是否成功,true为成功,false操作失败 + boolean success(); + + //操作代码 + int code(); + + //提示信息 + String message();*/ + +} diff --git a/src/main/java/com/msdw/tms/entity/vo/CdCreatePhotoVo.java b/src/main/java/com/msdw/tms/entity/vo/CdCreatePhotoVo.java new file mode 100644 index 0000000..886e8d9 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/vo/CdCreatePhotoVo.java @@ -0,0 +1,19 @@ +package com.msdw.tms.entity.vo; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +@Data +public class CdCreatePhotoVo { + + + //生成图片1 + @ApiModelProperty(value = "汇率价格走势图") + private String photo_url1; + //生成图片2 + @ApiModelProperty(value = "汇率损益分布图") + private String photo_url2; + //Var值 + @ApiModelProperty(value = "Var值") + private String var; +} diff --git a/src/main/java/com/msdw/tms/entity/vo/CdExperimentalEntityVo.java b/src/main/java/com/msdw/tms/entity/vo/CdExperimentalEntityVo.java new file mode 100644 index 0000000..e78b3f5 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/vo/CdExperimentalEntityVo.java @@ -0,0 +1,59 @@ +package com.msdw.tms.entity.vo; + +import cn.afterturn.easypoi.excel.annotation.Excel; +import cn.afterturn.easypoi.excel.annotation.ExcelTarget; +import com.baomidou.mybatisplus.annotation.TableName; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.io.Serializable; + +/** + * @描述:实验数据 + * @作者: Rong + * @日期: 2020-10-20 + */ +@Data +@ApiModel(value = "蒙特卡洛实验数据封装") +@TableName("cd_experimental") +@ExcelTarget("cd_experimental") +public class CdExperimentalEntityVo implements Serializable { + + private static final long serialVersionUID = 1L; + + @Excel(name = "编号", orderNum = "1", mergeVertical = true, isImportField = "id", width = 15) + @ApiModelProperty(value = "主键") + private Integer id; + + @Excel(name = "汇率类型", orderNum = "2", mergeVertical = true, isImportField = "exchangeRate", width = 15) + @ApiModelProperty(value = "汇率类型") + private String exchangeRate; + + @Excel(name = "样本数据", orderNum = "3", mergeVertical = true, isImportField = "sampleData", width = 25) + @ApiModelProperty(value = "样本数据") + private String sampleData; + + @Excel(name = "置信水平", orderNum = "4", mergeVertical = true, isImportField = "confidenceLevel", width = 20) + @ApiModelProperty(value = "置信水平") + private String confidenceLevel; + + @Excel(name = "持有天数", orderNum = "5", mergeVertical = true, isImportField = "validPeriod", width = 30) + @ApiModelProperty(value = "持有天数") + private String validPeriod; + + + @Excel(name = "汇率走势图", orderNum = "6", mergeVertical = true, isImportField = "trendChart", width = 30) + @ApiModelProperty(value = "汇率走势图") + private String trendChart; + + @Excel(name = "汇率损益分布图", orderNum = "7", mergeVertical = true, isImportField = "profitLossDistribution", width = 30) + @ApiModelProperty(value = "汇率损益分布图") + private String profitLossDistribution; + + @Excel(name = "创建时间", orderNum = "8", mergeVertical = true, isImportField = "creatTime", width = 30) + @ApiModelProperty(value = "创建时间") + private String creatTime; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/entity/vo/CdFuturesConfigureEntityVo.java b/src/main/java/com/msdw/tms/entity/vo/CdFuturesConfigureEntityVo.java new file mode 100644 index 0000000..03bfa38 --- /dev/null +++ b/src/main/java/com/msdw/tms/entity/vo/CdFuturesConfigureEntityVo.java @@ -0,0 +1,73 @@ +package com.msdw.tms.entity.vo; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +/** + * @描述:外汇期货平仓绩效 + * @作者: Rong + * @日期: 2020-10-12 + */ +@Data +@ApiModel(value = "外汇期货平仓绩效") +public class CdFuturesConfigureEntityVo { + + + //序号 + @TableId(type = IdType.AUTO) + private Integer id; + + //代码 + @ApiModelProperty(value = "代码") + private String eName; + + //合约名称 + @ApiModelProperty(value = "名称") + private String name; + + //头寸部位 + @ApiModelProperty(value = "交易方向(0空头,1多头)") + private String tradingDirection; + + //开仓价格(委托价格) + @ApiModelProperty(value = "开仓价格/委托价格") + private BigDecimal entrustPrice; + + + //平仓价格 + @ApiModelProperty(value = "平仓价格") + private BigDecimal closeRate; + + //资产市值1 + @ApiModelProperty(value = "资产市值1") + private BigDecimal marketValue1; + + //资产市值2 + @ApiModelProperty(value = "资产市值2") + private BigDecimal marketValue2; + + //数量(委托数量) + @ApiModelProperty(value = "数量(委托数量)") + private Integer entrustNumber; + + //盈亏 + @ApiModelProperty(value = "盈亏") + private BigDecimal wh_pl; + + + @ApiModelProperty(value = "开仓时间") + private String createTime; + + @ApiModelProperty(value = "平仓时间") + private String updateTime; + + @ApiModelProperty(value = "最后交易日") + private String tradeTime; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/feign/UserEntityFeignService.java b/src/main/java/com/msdw/tms/feign/UserEntityFeignService.java new file mode 100644 index 0000000..663b2c9 --- /dev/null +++ b/src/main/java/com/msdw/tms/feign/UserEntityFeignService.java @@ -0,0 +1,13 @@ +package com.msdw.tms.feign; + +import org.springframework.cloud.openfeign.FeignClient; +import com.msdw.tms.common.utils.R; +import org.springframework.web.bind.annotation.RequestMapping; + +@FeignClient +public interface UserEntityFeignService { + + @RequestMapping("userInfo/queryUserInfo") + public R getUser(); + +} diff --git a/src/main/java/com/msdw/tms/interceptor/AuthorizedAspect.java b/src/main/java/com/msdw/tms/interceptor/AuthorizedAspect.java new file mode 100644 index 0000000..8bd1976 --- /dev/null +++ b/src/main/java/com/msdw/tms/interceptor/AuthorizedAspect.java @@ -0,0 +1,92 @@ +package com.msdw.tms.interceptor; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.interceptor + * @ClassName: AuthorizedAspect + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/22 10:01 + * @UpdateDate: 2020/10/22 10:01 + * @Version: 1.0 + */ + +import org.apache.commons.lang.StringUtils; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.redis.core.StringRedisTemplate; +import org.springframework.stereotype.Component; +import org.springframework.web.servlet.HandlerInterceptor; +import org.springframework.web.servlet.ModelAndView; + +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; + + +@Component +public class AuthorizedAspect implements HandlerInterceptor { + @Autowired + private StringRedisTemplate redisTemplate; + + + @Override + public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { + //去redis里查询 + String token = request.getHeader("token"); + + String tokenValue = redisTemplate.opsForValue().get("token:" + token); + if (StringUtils.isEmpty(tokenValue)) { + System.err.println("【登录校验】Redis中查不到token"); + return false; + } + return true; + } + + @Override + public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { + + } + + @Override + public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { + + } + + + + /*@Override + public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { + //去redis里查询 + String tokenValue = redisTemplate.opsForValue().get("token"); + if (StringUtils.isEmpty(tokenValue)) { + log.warn("【登录校验】Redis中查不到token"); + return false; + } + return true; + }*/ + + + //切面拦截器,拦截订单接口请求中是否能查询到token + /* @Pointcut("execution(public * com.msdw.tms.controller.*(..))") //切入点 + public void verify() { + } + + @Before("verify()") //在请求之前 + public R doVerify() { + AuthCustomize authCustomize; + ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes(); + HttpServletRequest request = attributes.getRequest(); + String token = request.getHeader("token"); + + //去redis里查询 + String tokenValue = redisTemplate.opsForValue().get("token:" + token); + if (StringUtils.isEmpty(tokenValue)) { + log.warn("【登录校验】Redis中查不到token"); + return R.error("Redis中查不到token"); + } + return R.ok(); + }*/ + + + + +} diff --git a/src/main/java/com/msdw/tms/model/req/CdBuyInReq.java b/src/main/java/com/msdw/tms/model/req/CdBuyInReq.java new file mode 100644 index 0000000..2cc2b75 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdBuyInReq.java @@ -0,0 +1,21 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +@ApiModel(value = "平仓请求参数") +@Data +public class CdBuyInReq { + /*@ApiImplicitParams({ + @ApiImplicitParam(name = "id", value = "要平仓的序号", required = true, dataType = "string", paramType = "query"), + @ApiImplicitParam(name = "date", value = "行情滚动的日期", required = true, dataType = "string", paramType = "query"), + })*/ + + @ApiModelProperty("要平仓的序号") + private int id; + + @ApiModelProperty(value = "行情滚动的日期") + private String date; + +} diff --git a/src/main/java/com/msdw/tms/model/req/CdExperimentReq.java b/src/main/java/com/msdw/tms/model/req/CdExperimentReq.java new file mode 100644 index 0000000..3971d43 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdExperimentReq.java @@ -0,0 +1,62 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.req + * @ClassName: CdExperimentReq + * @Description: 实验保存 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 15:41 + * @UpdateDate: 2020/10/23 15:41 + * @Version: 1.0 + */ +@Data +@ApiModel(value = "实验保存") +public class CdExperimentReq { + /* + @ApiImplicitParams({ + @ApiImplicitParam(name = "name", value = "步骤二选择的币种", required = true, dataType = "String", paramType = "query"), + @ApiImplicitParam(name = "sampleData", value = "样本数据", required = true, dataType = "String", paramType = "query"), + @ApiImplicitParam(name = "confidenceLevel", value = "置信水平", required = true, dataType = "String", paramType = "query"), + @ApiImplicitParam(name = "validPeriod", value = "持有天数", required = false, dataType = "String", paramType = "query"), + @ApiImplicitParam(name = "varCode", value = "Var值", required = false, dataType = "String", paramType = "query"), + @ApiImplicitParam(name = "trendChart", value = "汇率价格走势图", required = false, dataType = "String", paramType = "query"), + @ApiImplicitParam(name = "profitLossDistribution", value = "汇率损益分布图", required = false, dataType = "String", paramType = "query"), + @ApiImplicitParam(name = "experimentsNumber", value = "实验次数", required = true, dataType = "String", paramType = "query"), + @ApiImplicitParam(name = "type", value = "实验类型(0正态Var,1历史模拟,2蒙特卡洛)", required = true, dataType = "String", paramType = "query") + }) + */ + + @ApiModelProperty(value = "步骤二选择的币种") + private String name; + + @ApiModelProperty(value = "样本数据") + private String sampleData; + + @ApiModelProperty(value = "置信水平") + private String confidenceLevel; + + @ApiModelProperty(value = "持有天数") + private String validPeriod; + + @ApiModelProperty(value = "Var值") + private String varCode; + + @ApiModelProperty(value = "汇率价格走势图") + private String trendChart; + + @ApiModelProperty(value = "汇率损益分布图") + private String profitLossDistribution; + + @ApiModelProperty(value = "实验类型(0正态Var,1历史模拟,2蒙特卡洛)") + private String type; + + @ApiModelProperty(value = "实验次数") + private String experimentsNumber; + + +} diff --git a/src/main/java/com/msdw/tms/model/req/CdExportReq.java b/src/main/java/com/msdw/tms/model/req/CdExportReq.java new file mode 100644 index 0000000..ab77e57 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdExportReq.java @@ -0,0 +1,31 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.mode.req + * @ClassName: ExportReq + * @Description: 导出 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 13:47 + * @UpdateDate: 2020/10/23 13:47 + * @Version: 1.0 + */ +@Data +@ApiModel("汇率中间价/根据时间、币种条件查询后导出报表") +public class CdExportReq { + /** + * @ApiParam(value = "要筛选的币种") @RequestParam(value = "name", required = true) String name, + * @ApiParam(value = "要筛选的开始时间") @RequestParam(value = "startTime", required = true) String startTime, + * @ApiParam(value = "要筛选的结束时间") @RequestParam(value = "endTime", required = true) String endTime + */ + @ApiModelProperty(value = "要筛选的币种") + public String name; + @ApiModelProperty(value = "要筛选的开始时间") + public String startTime; + @ApiModelProperty(value = "要筛选的结束时间") + public String endTime; +} diff --git a/src/main/java/com/msdw/tms/model/req/CdGetAccountInfoReq.java b/src/main/java/com/msdw/tms/model/req/CdGetAccountInfoReq.java new file mode 100644 index 0000000..7b27a8d --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdGetAccountInfoReq.java @@ -0,0 +1,40 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.NotNull; +import java.math.BigDecimal; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.req + * @ClassName: CdGetAccountInfoReq + * @Description: 期权账户请求参数 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/29 11:06 + * @UpdateDate: 2020/10/29 11:06 + * @Version: 1.0 + */ +@Data +@ApiModel("期权账户请求参数") +public class CdGetAccountInfoReq { + + @ApiModelProperty(value = "委托价格") + @NotNull(message = "委托价格不能为空") + private BigDecimal entrustPrice; + + @ApiModelProperty(value = "委托数量") + @NotNull(message = "委托数量不能为空") + private Integer entrustNumber; + + @ApiModelProperty(value = "交易方向(0空头,1多头)") + @NotBlank(message = "交易方向不能为空") + private Integer tradingDirection; + + @ApiModelProperty(value = "合约id") + @NotBlank(message = "合约id为空") + private Integer id; +} diff --git a/src/main/java/com/msdw/tms/model/req/CdGetCapitalReq.java b/src/main/java/com/msdw/tms/model/req/CdGetCapitalReq.java new file mode 100644 index 0000000..db7bbfe --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdGetCapitalReq.java @@ -0,0 +1,11 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +@Data +public class CdGetCapitalReq { + + @ApiModelProperty(value = "账户类型(1.银行账户 2.期货账户)") + private Integer transferOut; +} diff --git a/src/main/java/com/msdw/tms/model/req/CdGetWeekAndMonthInfoReq.java b/src/main/java/com/msdw/tms/model/req/CdGetWeekAndMonthInfoReq.java new file mode 100644 index 0000000..581f80c --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdGetWeekAndMonthInfoReq.java @@ -0,0 +1,23 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.req + * @ClassName: CdGetWeekAndMonthInfoReq + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/16 14:34 + * @UpdateDate: 2020/11/16 14:34 + * @Version: 1.0 + */ +@Data +public class CdGetWeekAndMonthInfoReq { + @ApiModelProperty("选择的id") + private int id; + + @ApiModelProperty(value = "选择的时间日期") + private String date; +} diff --git a/src/main/java/com/msdw/tms/model/req/CdOpenPositionReq.java b/src/main/java/com/msdw/tms/model/req/CdOpenPositionReq.java new file mode 100644 index 0000000..c66a14a --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdOpenPositionReq.java @@ -0,0 +1,41 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.NotNull; +import java.math.BigDecimal; + +@Data +@ApiModel("开仓请求参数") +public class CdOpenPositionReq { + private static final long serialVersionUID = 1L; + + + @ApiModelProperty(value = "合约表主键") + @NotNull(message = "合约表主键不能为空") + private Integer pid; + + @ApiModelProperty(value = "交易方向(0空头,1多头)") + @NotBlank(message = "交易方向不能为空") + private Integer tradingDirection; + + @ApiModelProperty(value = "最新价格") + @NotBlank(message = "最新价格不能为空") + private BigDecimal newestPrice; + + @ApiModelProperty(value = "委托价格") + @NotNull(message = "委托价格不能为空") + private BigDecimal entrustPrice; + + @ApiModelProperty(value = "委托数量") + @NotNull(message = "委托数量不能为空") + private Integer entrustNumber; + + @ApiModelProperty(value = "所需资金") + private BigDecimal fundingNeeded; + + +} diff --git a/src/main/java/com/msdw/tms/model/req/CdPageConditionReq.java b/src/main/java/com/msdw/tms/model/req/CdPageConditionReq.java new file mode 100644 index 0000000..83683cd --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdPageConditionReq.java @@ -0,0 +1,38 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.mode.req + * @ClassName: pageConditionReq + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 13:50 + * @UpdateDate: 2020/10/23 13:50 + * @Version: 1.0 + */ +@Data +public class CdPageConditionReq { + /** + * @ApiImplicitParams({ + * @ApiImplicitParam(name = "pageNum", value = "pageIndex:当前页数"), + * @ApiImplicitParam(name = "pageSize", value = "pageSize:当前页需要显示的数量"), + * @ApiImplicitParam(name = "name", value = "要筛选的币种"), + * @ApiImplicitParam(name = "date", value = "要筛选的时间") + * }) + */ + + @ApiModelProperty(value = "当前页数") + public Integer pageNum; + + @ApiModelProperty(value = "当前页需要显示的数量") + public Integer pageSize; + + @ApiModelProperty(value = "要筛选的币种") + public String name; + + @ApiModelProperty(value = "要筛选的结束时间") + public String date; +} diff --git a/src/main/java/com/msdw/tms/model/req/CdSaveCacheReq.java b/src/main/java/com/msdw/tms/model/req/CdSaveCacheReq.java new file mode 100644 index 0000000..d21da8a --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdSaveCacheReq.java @@ -0,0 +1,25 @@ +package com.msdw.tms.model.req; + +import com.alibaba.fastjson.JSONObject; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.req + * @ClassName: SaveCacheRwq + * @Description: 缓存步骤请求参数 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 15:24 + * @UpdateDate: 2020/10/23 15:24 + * @Version: 1.0 + */ +@Data +public class CdSaveCacheReq { + + @ApiModelProperty(value = "当前步骤数(传数字)") + private Integer stepNumber; + + @ApiModelProperty(value = "json字符串") + private JSONObject json; +} diff --git a/src/main/java/com/msdw/tms/model/req/CdSimulationReq.java b/src/main/java/com/msdw/tms/model/req/CdSimulationReq.java new file mode 100644 index 0000000..30989f2 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdSimulationReq.java @@ -0,0 +1,35 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.mode.req + * @ClassName: SimulationReq + * @Description: 根据选择的模拟法传值获取相应数据 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 13:29 + * @UpdateDate: 2020/10/23 13:29 + * @Version: 1.0 + */ +@Data +public class CdSimulationReq { + + @ApiModelProperty(value = "类型(0.正态VaR 1.历史模拟法 2.蒙特卡洛模拟法) 传数字") + private Integer type; + @ApiModelProperty(value = "要筛选的币种") + private String name; + @ApiModelProperty(value = "样本区间开始时间") + private String startTime; + @ApiModelProperty(value = "样本区间结束时间:由第二步骤中选择的日期") + private String endTime; + @ApiModelProperty(value = "置信水平") + private String confidenceLevel; + @ApiModelProperty(value = "持有效") + private String validity; + @ApiModelProperty(value = "实验次数") + private String experimentsNumber; + + +} diff --git a/src/main/java/com/msdw/tms/model/req/CdTransferRecordsReq.java b/src/main/java/com/msdw/tms/model/req/CdTransferRecordsReq.java new file mode 100644 index 0000000..2908091 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdTransferRecordsReq.java @@ -0,0 +1,39 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.req + * @ClassName: CdTransferRecordsReq + * @Description: 转账记录请求参数 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 16:30 + * @UpdateDate: 2020/10/23 16:30 + * @Version: 1.0 + */ +@Data +@ApiModel(value = "转账记录", description = "查询对象封装") +public class CdTransferRecordsReq { + /** + * @ApiImplicitParams({ + * @ApiImplicitParam(name = "pageNum", value = "pageIndex:当前页数", required = true, dataType = "String", paramType = "query"), + * @ApiImplicitParam(name = "pageSize", value = "pageSize:当前页需要显示的数量", required = true, dataType = "String", paramType = "query"), + * @ApiImplicitParam(name = "startTime", value = "开始时间", required = false, dataType = "String", paramType = "query"), + * @ApiImplicitParam(name = "endTime", value = "结束时间", required = false, dataType = "String", paramType = "query") + * }) + */ + + @ApiModelProperty(value = "开始时间") + private String startTime; + @ApiModelProperty(value = "结束时间") + private String endTime; + @ApiModelProperty(value = "当前页数") + private Integer pageNum; + @ApiModelProperty(value = "当前页需要显示的数量") + private Integer pageSize; + + +} diff --git a/src/main/java/com/msdw/tms/model/req/CdTransferReq.java b/src/main/java/com/msdw/tms/model/req/CdTransferReq.java new file mode 100644 index 0000000..04a1b7e --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdTransferReq.java @@ -0,0 +1,22 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +@Data +public class CdTransferReq { + /** + * @ApiImplicitParams({ + * @ApiImplicitParam(name = "transferOut", value = "要转出的账户(1.银行账户 2.期货账户)", required = true, dataType = "string", paramType = "query"), + * @ApiImplicitParam(name = "amountMoney", value = "转账金额", required = true, dataType = "string", paramType = "query"), + * }) + */ + + @ApiModelProperty("要转出的账户(1.银行账户 2.期货账户)") + private Integer transferOut; + + @ApiModelProperty("转账金额") + private BigDecimal amountMoney; +} diff --git a/src/main/java/com/msdw/tms/model/req/CdTransferWarehouseReq.java b/src/main/java/com/msdw/tms/model/req/CdTransferWarehouseReq.java new file mode 100644 index 0000000..33b9855 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/CdTransferWarehouseReq.java @@ -0,0 +1,37 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotNull; +import java.math.BigDecimal; + +@Data +@ApiModel("调仓请求参数") +public class CdTransferWarehouseReq { + + @ApiModelProperty(value = "主键", required = false) + private Integer id; + + /* @ApiModelProperty(value = "交易方向(0空头,1多头)") + @NotBlank(message = "交易方向不能为空") + private Integer tradingDirection;*/ + + @ApiModelProperty(value = "买卖方向:0.开仓 1.平仓") + private Integer status; + + @ApiModelProperty(value = "委托价格") + @NotNull(message = "委托价格不能为空") + private BigDecimal entrustPrice; + + @ApiModelProperty(value = "委托手数") + @NotNull(message = "委托数量不能为空") + private Integer entrustNumber; + + @ApiModelProperty(value = "可交易数量") + @NotNull(message = "可交易数量不能为空") + private BigDecimal tradableNumber; + + +} diff --git a/src/main/java/com/msdw/tms/model/req/QhscGangjiaosuoReq.java b/src/main/java/com/msdw/tms/model/req/QhscGangjiaosuoReq.java new file mode 100644 index 0000000..ecb274c --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/QhscGangjiaosuoReq.java @@ -0,0 +1,38 @@ +package com.msdw.tms.model.req; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotEmpty; + +@Data +public class QhscGangjiaosuoReq { + + //String name, String date, Integer pageNum, Integer pageSize + + /** + * @ApiImplicitParams({ + * @ApiImplicitParam(name = "pageNum", value = "pageIndex:当前页数", required = true, dataType = "String", paramType = "query"), + * @ApiImplicitParam(name = "pageSize", value = "pageSize:当前页需要显示的数量", required = true, dataType = "String", paramType = "query"), + * @ApiImplicitParam(name = "name", value = "要筛选的币种", required = false, dataType = "String", paramType = "query"), + * @ApiImplicitParam(name = "date", value = "日期(步骤二选中的时间)", required = false, dataType = "String", paramType = "query") + * }) + */ + + @ApiModelProperty(value = "要筛选的币种") + private String name; + + @ApiModelProperty(value = "步骤二选中的时间") + @NotEmpty(message = "时间不能为空") + private String date; + + @ApiModelProperty(value = "当前页数") + @NotEmpty(message = "当前页数不能为空") + private Integer pageNum; + + @ApiModelProperty(value = "当前页需要显示的数量") + @NotEmpty(message = "当前页需要显示的数量不能为空") + private Integer pageSize; + + +} diff --git a/src/main/java/com/msdw/tms/model/req/Testreq.java b/src/main/java/com/msdw/tms/model/req/Testreq.java new file mode 100644 index 0000000..2905c16 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/req/Testreq.java @@ -0,0 +1,14 @@ +package com.msdw.tms.model.req; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.mode.req + * @ClassName: Testreq + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 10:56 + * @UpdateDate: 2020/10/23 10:56 + * @Version: 1.0 + */ +public class Testreq { +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdAccountInfoResp.java b/src/main/java/com/msdw/tms/model/resp/CdAccountInfoResp.java new file mode 100644 index 0000000..4e14ecc --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdAccountInfoResp.java @@ -0,0 +1,67 @@ +package com.msdw.tms.model.resp; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.req + * @ClassName: CdAccountInfoResp + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/26 11:13 + * @UpdateDate: 2020/10/26 11:13 + * @Version: 1.0 + */ +@ApiModel(value = "期货期权账户返回参数") +@Data +public class CdAccountInfoResp { + + @ApiModelProperty(value = "总资产") + private BigDecimal totalAssets; + + @ApiModelProperty(value = "可用资金") + private BigDecimal balance; + + @ApiModelProperty(value = "资产市值") + private BigDecimal marketAssets; + + @ApiModelProperty(value = "浮动盈亏") + private BigDecimal floatingPL; + + @ApiModelProperty(value = "占用保证金") + private BigDecimal depositBalance; + + @ApiModelProperty(value = "冻结资金") + private BigDecimal forzenBalance; + + + /** + * 待开仓信息 + */ + @ApiModelProperty(value = "序号") + private Integer id; + + @ApiModelProperty(value = "合约名称") + private transient String name; + + @ApiModelProperty(value = "代码") + private transient String ename; + + @ApiModelProperty(value = "交易方向(0空头,1多头)") + private Integer tradingDirection; + + @ApiModelProperty(value = "创建时间") + private String createTime; + + @ApiModelProperty(value = "最后交易日") + private String tradeTime; + + @ApiModelProperty(value = "所需资金") + private BigDecimal fundingNeeded; + + +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdAfterHedgingResp.java b/src/main/java/com/msdw/tms/model/resp/CdAfterHedgingResp.java new file mode 100644 index 0000000..6662e55 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdAfterHedgingResp.java @@ -0,0 +1,45 @@ +package com.msdw.tms.model.resp; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.resp + * @ClassName: CdAfterHedgingResp + * @Description: 对冲后返回对象 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/30 9:17 + * @UpdateDate: 2020/10/30 9:17 + * @Version: 1.0 + */ +@Data +@ApiModel("对冲后损益情况返回信息") +public class CdAfterHedgingResp { + + @ApiModelProperty(value = "汇率类型") + private String name; + + @ApiModelProperty(value = "样本区间:开始时间") + private String startTime; + + @ApiModelProperty(value = "样本区间:结束时间") + private String endTime; + + @ApiModelProperty(value = "持有效") + private String effective; + + @ApiModelProperty("置信水平") + private String confidenceLevel; + + @ApiModelProperty("实验次数") + private String experimentNum; + + @ApiModelProperty("对冲前损益分布图") + private String beforeHedgingUrl; + + @ApiModelProperty("对冲后损益分布图") + private String afterHedgingUrl; + +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdBeforeHedgingResp.java b/src/main/java/com/msdw/tms/model/resp/CdBeforeHedgingResp.java new file mode 100644 index 0000000..efd0f26 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdBeforeHedgingResp.java @@ -0,0 +1,73 @@ +package com.msdw.tms.model.resp; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.resp + * @ClassName: CdBeforeHedgingResp + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/28 14:26 + * @UpdateDate: 2020/10/28 14:26 + * @Version: 1.0 + */ +@ApiModel(value = "返回信息", description = "外汇现货盈亏返回对象") +@Data +public class CdBeforeHedgingResp { + + + @ApiModelProperty(value = "序号", required = false) + private Integer id; + + @ApiModelProperty(value = "交易方向(0空头卖,1多头买)") + private String tradingDirection; + + @ApiModelProperty(value = "(持仓可用)委托数量") + private Integer entrustNumber; + + @ApiModelProperty(value = "持仓状态(0.开仓 1.平仓)") + private String status; + + @ApiModelProperty(value = "开仓时间") + private String createTime; + + @ApiModelProperty(value = "平仓时间") + private String updateTime; + + @ApiModelProperty(value = "合约名称") + private String name; + + @ApiModelProperty(value = "开仓价格") + private BigDecimal entrustPrice; + + @ApiModelProperty(value = "平仓价格") + private BigDecimal closeRate; + + @ApiModelProperty(value = "金额外币(开仓)") + private BigDecimal foreignCurrencyByOpen; + + @ApiModelProperty(value = "金额外币(平仓)") + private BigDecimal foreignCurrencyByClose; + + @ApiModelProperty(value = "所需资金(开仓)") + private BigDecimal neededPriceByOpen; + + @ApiModelProperty(value = "所需资金(平仓)") + private BigDecimal neededPriceByClose; + + @ApiModelProperty(value = "盈亏") + private BigDecimal profitLoss; + + @ApiModelProperty(value = "汇率1") + private BigDecimal exchangeRate1; + + @ApiModelProperty(value = "汇率2") + private BigDecimal exchangeRate2; + + +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdFxSpotResp.java b/src/main/java/com/msdw/tms/model/resp/CdFxSpotResp.java new file mode 100644 index 0000000..389035d --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdFxSpotResp.java @@ -0,0 +1,83 @@ +package com.msdw.tms.model.resp; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import javax.validation.constraints.NotBlank; +import javax.validation.constraints.NotNull; +import java.math.BigDecimal; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.req + * @ClassName: CdFxSpotReq + * @Description: 外汇现货返回类 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/27 9:19 + * @UpdateDate: 2020/10/27 9:19 + * @Version: 1.0 + */ +@Data +public class CdFxSpotResp { + + @ApiModelProperty(value = "序号", required = false) + @TableId(type = IdType.AUTO) + private Integer id; + + @ApiModelProperty(value = "用户id") + @NotNull(message = "用户id不能为空") + private Integer userId; + + @ApiModelProperty(value = "合约表主键") + @NotNull(message = "合约表主键不能为空") + private Integer pid; + + @ApiModelProperty(value = "交易方向(0空头,1多头)") + @NotBlank(message = "交易方向不能为空") + private Integer tradingDirection; + + @ApiModelProperty(value = "最新价格") + @NotBlank(message = "最新价格不能为空") + private BigDecimal newestPrice; + + @ApiModelProperty(value = "委托价格") + @NotNull(message = "委托价格不能为空") + private BigDecimal entrustPrice; + + @ApiModelProperty(value = "委托数量") + @NotNull(message = "委托数量不能为空") + private Integer entrustNumber; + + @ApiModelProperty(value = "可交易数量") + @NotNull(message = "可交易数量不能为空") + private BigDecimal tradableNumber; + + + @ApiModelProperty(value = "0.开仓 1.平仓") + private Integer status; + + @ApiModelProperty(value = "是否自动平仓(0否 1是)") + private Integer type; + + @ApiModelProperty(value = "浮动盈亏") + private BigDecimal floatingPl; + + @ApiModelProperty(value = "开仓时间") + @NotBlank(message = "开仓时间不能为空") + private String createTime; + + @ApiModelProperty(value = "平仓时间") + @NotBlank(message = "平仓时间不能为空") + private String updateTime; + + @ApiModelProperty(value = "最后交易日") + @NotBlank(message = "最后交易日不能为空") + private String tradeTime; + + + @ApiModelProperty(value = "平仓价格") + @NotBlank(message = "平仓价格不能为空") + private BigDecimal closeRate; +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdGetDetailByIdResp.java b/src/main/java/com/msdw/tms/model/resp/CdGetDetailByIdResp.java new file mode 100644 index 0000000..c8dad3d --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdGetDetailByIdResp.java @@ -0,0 +1,54 @@ +package com.msdw.tms.model.resp; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.resp + * @ClassName: CdGetDetailByIdResp + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/27 18:45 + * @UpdateDate: 2020/10/27 18:45 + * @Version: 1.0 + */ +@Data +public class CdGetDetailByIdResp { + + @ApiModelProperty(value = "合约表主键") + private Integer pid; + @ApiModelProperty(value = "合约名称") + private String name; + + @ApiModelProperty(value = "最新价格") + private BigDecimal newestPrice; + + @ApiModelProperty(value = "委托价格(暂默认为最新价格)") + private BigDecimal entrustPrice; + + @ApiModelProperty(value = "委托数量(这里默认为1)") + private Integer entrustNumber; + + @ApiModelProperty(value = "可交易数量") + private String tradableNumber; + + @ApiModelProperty(value = "预冻结保证金") + private BigDecimal preFreezingMargin; + + @ApiModelProperty(value = "预冻结手续费") + private BigDecimal preFreezingFrozen; + + @ApiModelProperty(value = "冻结资金") + private BigDecimal frozenPrice; + + @ApiModelProperty(value = "交易单位") + private Integer tradingUnit; + + @ApiModelProperty(value = "交易保证金") + private BigDecimal tradingBond; + + +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdGetExchangeRateResp.java b/src/main/java/com/msdw/tms/model/resp/CdGetExchangeRateResp.java new file mode 100644 index 0000000..5f98f47 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdGetExchangeRateResp.java @@ -0,0 +1,21 @@ +package com.msdw.tms.model.resp; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.resp + * @ClassName: CdGetExchangeRateResp + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/27 10:21 + * @UpdateDate: 2020/10/27 10:21 + * @Version: 1.0 + */ +@Data +public class CdGetExchangeRateResp { + //汇率类型 + @ApiModelProperty(value = "汇率类型") + private String typeName; +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdGetWeekAndMonthInfoResp.java b/src/main/java/com/msdw/tms/model/resp/CdGetWeekAndMonthInfoResp.java new file mode 100644 index 0000000..a1255be --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdGetWeekAndMonthInfoResp.java @@ -0,0 +1,29 @@ +package com.msdw.tms.model.resp; + +import com.baomidou.mybatisplus.annotation.TableField; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.resp + * @ClassName: CdGetWeekAndMonthInfoResp + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/16 18:32 + * @UpdateDate: 2020/11/16 18:32 + * @Version: 1.0 + */ +@Data +@ApiModel(value = "查看外汇每周/每月的信息返回对象") +public class CdGetWeekAndMonthInfoResp { + @TableField("last_price") + @ApiModelProperty(value = "最新价") + private Double lastPrice; + + + @TableField("date") + @ApiModelProperty(value = "时间") + private String date; +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdHedgingEfficiencyResp.java b/src/main/java/com/msdw/tms/model/resp/CdHedgingEfficiencyResp.java new file mode 100644 index 0000000..5bf8f27 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdHedgingEfficiencyResp.java @@ -0,0 +1,53 @@ +package com.msdw.tms.model.resp; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.resp + * @ClassName: CdHedgingEfficiencyResp + * @Description: 对冲效率 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/3 12:02 + * @UpdateDate: 2020/11/3 12:02 + * @Version: 1.0 + */ +@Data +@ApiModel("对冲效率") +public class CdHedgingEfficiencyResp { + //序号 + @ApiModelProperty(value = "序号") + private Integer id; + + //合约名称 + @ApiModelProperty(value = "合约名称") + private String name; + + //现货盈亏 + @ApiModelProperty(value = "现货盈亏") + private BigDecimal promptGoodsPL; + + //期货盈亏 + @ApiModelProperty(value = "期货盈亏") + private BigDecimal futuresPl; + + //总盈亏 + @ApiModelProperty(value = "总盈亏") + private BigDecimal totalPl; + //总盈亏率 + @ApiModelProperty(value = "总盈亏率") + private String totalPlRatio; + + //对冲风险效率 + @ApiModelProperty(value = "对冲风险效率") + private String riskEfficiency; + + //对冲风险比率 + @ApiModelProperty(value = "对冲风险比率") + private String riskRatio; + +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdHedgingResp.java b/src/main/java/com/msdw/tms/model/resp/CdHedgingResp.java new file mode 100644 index 0000000..5a9ef68 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdHedgingResp.java @@ -0,0 +1,33 @@ +package com.msdw.tms.model.resp; + +import com.msdw.tms.entity.vo.CdFuturesConfigureEntityVo; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.resp + * @ClassName: CdHedgingResp + * @Description: 对冲信息返回 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/3 10:07 + * @UpdateDate: 2020/11/3 10:07 + * @Version: 1.0 + */ +@Data +@ApiModel("对冲返回信息") +public class CdHedgingResp { + + + @ApiModelProperty(value = "外汇期货平仓绩效", required = false) + private CdFuturesConfigureEntityVo configure; + + @ApiModelProperty(value = "外汇现货盈亏", required = false) + private CdBeforeHedgingResp beforeHedging; + + @ApiModelProperty(value = "对冲效率", required = false) + private CdHedgingEfficiencyResp hedgingEfficiency; + + +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdHoldPositionInfoResp.java b/src/main/java/com/msdw/tms/model/resp/CdHoldPositionInfoResp.java new file mode 100644 index 0000000..1447f72 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdHoldPositionInfoResp.java @@ -0,0 +1,67 @@ +package com.msdw.tms.model.resp; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.resp + * @ClassName: CdHoldPositionInfoResp + * @Description: 外汇期货持仓信息 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/28 11:34 + * @UpdateDate: 2020/10/28 11:34 + * @Version: 1.0 + */ +@ApiModel(value = "返回信息", description = "外汇期货持仓信息") +@Data +public class CdHoldPositionInfoResp { + @ApiModelProperty(value = "序号", required = false) + private Integer id; + + @ApiModelProperty(value = "用户id") + private Integer userId; + + + @ApiModelProperty(value = "交易方向(0空头卖,1多头买)") + private String tradingDirection; + + + @ApiModelProperty(value = "均价/现价") + private BigDecimal entrustPrice; + + @ApiModelProperty(value = "(持仓可用)委托数量") + private Integer entrustNumber; + + @ApiModelProperty(value = "持仓状态(0.开仓 1.平仓)") + private String status; + + + @ApiModelProperty(value = "开仓时间") + private String createTime; + + @ApiModelProperty(value = "平仓时间") + private String updateTime; + + @ApiModelProperty(value = "最后交易日") + private String tradeTime; + + + @ApiModelProperty(value = "合约名称") + private String name; + + @ApiModelProperty(value = "合约代码") + private String ename; + + + @ApiModelProperty(value = "浮动盈亏") + private BigDecimal floatingPl; + + @ApiModelProperty(value = "保证金") + private BigDecimal preFreezingMargin; + + +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdTransferOutResp.java b/src/main/java/com/msdw/tms/model/resp/CdTransferOutResp.java new file mode 100644 index 0000000..95583e2 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdTransferOutResp.java @@ -0,0 +1,36 @@ +package com.msdw.tms.model.resp; + +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +@Data +public class CdTransferOutResp { + + @ApiModelProperty(value = "用户id") + private Long userId; + + @ApiModelProperty(value = "可用资金") + private BigDecimal balance; + + @ApiModelProperty(value = "冻结资金") + private BigDecimal forzenBalance; + + @ApiModelProperty(value = "占用保证金") + private BigDecimal depositBalance; + + @ApiModelProperty(value = "创建时间") + private String createTime; + + @ApiModelProperty(value = "修改时间") + private String updateTime; + + @ApiModelProperty(value = "初始资金") + private BigDecimal initialBalance; + + @ApiModelProperty(value = "总资产") + private transient BigDecimal totalAssets; + + +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdTransferRecordsResp.java b/src/main/java/com/msdw/tms/model/resp/CdTransferRecordsResp.java new file mode 100644 index 0000000..fc458ad --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdTransferRecordsResp.java @@ -0,0 +1,39 @@ +package com.msdw.tms.model.resp; + +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +@Data +@ApiModel(value = "查询对象", description = "转账记录查询对象封装") +public class CdTransferRecordsResp { + + //SELECT create_time,type,amount_money,coin_type from cd_bank_statements; + + /** + * 类型(1银行账户 2期货账户) + */ + @ApiModelProperty(value = "转出账户") + private String out; + + @ApiModelProperty(value = "转入账户") + private String enter; + + @ApiModelProperty(value = "币种") + private String coinType; + + + @ApiModelProperty(value = "金额") + private BigDecimal amountMoney; + + + @ApiModelProperty(value = "转账时间") + private String createTime; + + + @ApiModelProperty(value = "转账类型") + private String type; + +} diff --git a/src/main/java/com/msdw/tms/model/resp/CdTransferWarehouseDetailResp.java b/src/main/java/com/msdw/tms/model/resp/CdTransferWarehouseDetailResp.java new file mode 100644 index 0000000..9334c96 --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/CdTransferWarehouseDetailResp.java @@ -0,0 +1,86 @@ +package com.msdw.tms.model.resp; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import io.swagger.annotations.ApiModel; +import io.swagger.annotations.ApiModelProperty; +import lombok.Data; + +import java.math.BigDecimal; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.model.resp + * @ClassName: CdTransferWarehouseDetailResp + * @Description: 调仓展示详情 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/2 15:40 + * @UpdateDate: 2020/11/2 15:40 + * @Version: 1.0 + */ +@Data +@ApiModel("调仓前详情展示") +public class CdTransferWarehouseDetailResp { + + @ApiModelProperty(value = "主键", required = false) + @TableId(type = IdType.AUTO) + private Integer id; + + + @ApiModelProperty(value = "合约表主键") + private Integer pid; + + @ApiModelProperty(value = "交易方向(0空头,1多头)") + private Integer tradingDirection; + + @ApiModelProperty(value = "委托价") + private BigDecimal newestPrice; + + @ApiModelProperty(value = "名称") + private transient String name; + + @ApiModelProperty(value = "代码") + private transient String ename; + + @ApiModelProperty(value = "当前最新价") + private transient BigDecimal lastPrice; + + @ApiModelProperty(value = "涨跌幅") + private transient String u; + + @ApiModelProperty(value = "昨收") + private transient BigDecimal dayUp; + + @ApiModelProperty(value = "买卖方向为开仓时候:可交易数量为Python表的持仓总量") + private transient BigDecimal tradableQuantityByBuy; + + @ApiModelProperty(value = "买卖方向为平仓时候:可交易数量为自己的持仓总量") + private transient BigDecimal tradableQuantityBySell; + + @ApiModelProperty(value = "当前期权账户可用资金") + private transient BigDecimal availableFunds; + + + @ApiModelProperty(value = "涨跌额") + private BigDecimal udPrice; + + @ApiModelProperty(value = "买入价") + private BigDecimal buyPrice; + + @ApiModelProperty(value = "卖出价") + private BigDecimal cellPrice; + + @ApiModelProperty(value = "买量") + private Integer buy; + + @ApiModelProperty(value = "卖量") + private Integer cell; + + @ApiModelProperty(value = "总量") + private BigDecimal suoyou; + + @ApiModelProperty(value = "持仓量") + private String hava; + + +} diff --git a/src/main/java/com/msdw/tms/model/resp/TestResq.java b/src/main/java/com/msdw/tms/model/resp/TestResq.java new file mode 100644 index 0000000..258770b --- /dev/null +++ b/src/main/java/com/msdw/tms/model/resp/TestResq.java @@ -0,0 +1,14 @@ +package com.msdw.tms.model.resp; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.mode.resp + * @ClassName: TestResq + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/10/23 10:57 + * @UpdateDate: 2020/10/23 10:57 + * @Version: 1.0 + */ +public class TestResq { +} diff --git a/src/main/java/com/msdw/tms/service/AliyunOssService.java b/src/main/java/com/msdw/tms/service/AliyunOssService.java new file mode 100644 index 0000000..9833ba4 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/AliyunOssService.java @@ -0,0 +1,29 @@ +package com.msdw.tms.service; + +import com.msdw.tms.common.utils.FilesResult; +import org.springframework.web.multipart.MultipartFile; + +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; + +public interface AliyunOssService { + + /** + * 上传文件 + * + * @param file 上传文件 + */ + FilesResult uploadFiles(MultipartFile file) throws IOException; + + /** + * 下载文件 + * + * @param objectName 本地路径 + */ + void downloadFiles(HttpServletResponse response, String objectName) throws IOException; + + /** + * 根据文件路径+文件名称,删除该文件 + */ + void deleteFile(String filename); +} diff --git a/src/main/java/com/msdw/tms/service/CdBankStatementsService.java b/src/main/java/com/msdw/tms/service/CdBankStatementsService.java new file mode 100644 index 0000000..fb50e94 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/CdBankStatementsService.java @@ -0,0 +1,29 @@ +package com.msdw.tms.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.entity.CdBankStatementsEntity; +import com.msdw.tms.model.req.CdGetAccountInfoReq; +import com.msdw.tms.model.resp.CdTransferRecordsResp; + +import java.math.BigDecimal; +import java.util.List; + +/** + * @描述:账户流水记录 service接口 + * @作者: Rong + * @日期: 2020-10-15 + */ +public interface CdBankStatementsService extends IService { + + boolean modify(Integer userId, Integer transferOut, BigDecimal amountMoney); + + + //转账交易记录 + List queryPageByRecode(String userId, String startTime, String endTime); + + + //查看期权账户时候返回头寸配置信息 + R getAccountInfoReturnOpen(CdGetAccountInfoReq req, Integer userId); + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/CdCacheService.java b/src/main/java/com/msdw/tms/service/CdCacheService.java new file mode 100644 index 0000000..2da62be --- /dev/null +++ b/src/main/java/com/msdw/tms/service/CdCacheService.java @@ -0,0 +1,16 @@ +package com.msdw.tms.service; + +import com.alibaba.fastjson.JSONObject; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.entity.CdCacheEntity; +import com.baomidou.mybatisplus.extension.service.IService; + +/** + * @描述:步骤缓存记录表 service接口 + * @作者: Rong + * @日期: 2020-10-21 + */ +public interface CdCacheService extends IService { + R cache(Integer userId, Integer stepNumber, JSONObject json); + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/CdExperimentalService.java b/src/main/java/com/msdw/tms/service/CdExperimentalService.java new file mode 100644 index 0000000..e24a571 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/CdExperimentalService.java @@ -0,0 +1,28 @@ +package com.msdw.tms.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.entity.CdExperimentalEntity; + +import javax.servlet.http.HttpServletResponse; +import java.util.List; + +/** + * @描述:实验数据 service接口 + * @作者: Rong + * @日期: 2020-10-20 + */ + +public interface CdExperimentalService extends IService { + + //单条导出实验结果 + R exportProjectRecord(HttpServletResponse response, Integer id) throws Exception; + + //多条导出实验结果 + R exportexportProjectRecordByAll(HttpServletResponse response, Integer userId, Integer type) throws Exception; + + //根据选择的id导出实验结果 + R exportexportProjectRecordById(HttpServletResponse response, Integer userId, List ids) throws Exception; + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/CdFuturesConfigureService.java b/src/main/java/com/msdw/tms/service/CdFuturesConfigureService.java new file mode 100644 index 0000000..1652fd3 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/CdFuturesConfigureService.java @@ -0,0 +1,47 @@ +package com.msdw.tms.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.entity.CdFuturesConfigureEntity; +import com.msdw.tms.entity.vo.CdFuturesConfigureEntityVo; +import com.msdw.tms.model.resp.CdAfterHedgingResp; +import com.msdw.tms.model.resp.CdBeforeHedgingResp; +import com.msdw.tms.model.resp.CdHoldPositionInfoResp; +import com.msdw.tms.model.resp.CdTransferWarehouseDetailResp; +import org.springframework.stereotype.Service; + +/** + * @描述:期货资产头寸配置 service接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Service +public interface CdFuturesConfigureService extends IService { + //开仓 + R openPosition(CdFuturesConfigureEntity entity); + + //平仓 + R buyIn(int id, String date, Integer userId); + + //调仓详情展示 + CdTransferWarehouseDetailResp transferWarehouseByDetail(int id); + + + //外汇期货持仓情况 + CdHoldPositionInfoResp myPosition(int id); + + //外汇期货平仓绩效 + CdFuturesConfigureEntityVo buyInAchievements(int id, Integer userId); + + //外汇现货盈亏 + CdBeforeHedgingResp beforeHedging(int userId); + + + //对冲后损益情况 + CdAfterHedgingResp afterHedging(int userId); + + //根据id删除开仓信息 + boolean delOpenPosition(int id,int userId); + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/CdPointChildrenService.java b/src/main/java/com/msdw/tms/service/CdPointChildrenService.java new file mode 100644 index 0000000..0776af0 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/CdPointChildrenService.java @@ -0,0 +1,14 @@ +package com.msdw.tms.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.msdw.tms.entity.CdPointChildren; + +/** + * @描述:判分点值子级表 service接口 + * @作者: Rong + * @日期: 2021-01-12 + */ +public interface CdPointChildrenService extends IService { + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/CdPointService.java b/src/main/java/com/msdw/tms/service/CdPointService.java new file mode 100644 index 0000000..6794e38 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/CdPointService.java @@ -0,0 +1,14 @@ +package com.msdw.tms.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.msdw.tms.entity.CdPoint; + +/** + * @描述: service接口 + * @作者: Rong + * @日期: 2021-01-11 + */ +public interface CdPointService extends IService { + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/CdTransferWarehouseRecordService.java b/src/main/java/com/msdw/tms/service/CdTransferWarehouseRecordService.java new file mode 100644 index 0000000..88f7105 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/CdTransferWarehouseRecordService.java @@ -0,0 +1,17 @@ +package com.msdw.tms.service; + +import com.msdw.tms.common.utils.R; +import com.msdw.tms.entity.CdFuturesConfigureEntity; +import com.msdw.tms.entity.CdTransferWarehouseRecordEntity; +import com.baomidou.mybatisplus.extension.service.IService; +import org.springframework.stereotype.Service; + +/** + * @描述:调仓记录表 service接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Service +public interface CdTransferWarehouseRecordService extends IService { + R add(CdFuturesConfigureEntity entity); +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/CdUserAssetsService.java b/src/main/java/com/msdw/tms/service/CdUserAssetsService.java new file mode 100644 index 0000000..6ce9fbf --- /dev/null +++ b/src/main/java/com/msdw/tms/service/CdUserAssetsService.java @@ -0,0 +1,21 @@ +package com.msdw.tms.service; + +import com.msdw.tms.entity.CdUserAssetsEntity; +import com.baomidou.mybatisplus.extension.service.IService; + +/** + * @描述:用户银行账户 service接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +public interface CdUserAssetsService extends IService { + + int addInfo(CdUserAssetsEntity userAssetsEntity); + + //根据用户id查询银行账户信息 + CdUserAssetsEntity userAssetsByUerId(long userId); + + //根据用户id修改 + int modify(long userId); + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/CdUserOptionAccountService.java b/src/main/java/com/msdw/tms/service/CdUserOptionAccountService.java new file mode 100644 index 0000000..b7e9c72 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/CdUserOptionAccountService.java @@ -0,0 +1,21 @@ +package com.msdw.tms.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.msdw.tms.entity.CdUserOptionAccountEntity; + +/** + * @描述:期货期权账户 service接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +public interface CdUserOptionAccountService extends IService { + + //新增期权账户信息 + int addInfoByOptionAccount(CdUserOptionAccountEntity userOptionAccount); + + CdUserOptionAccountEntity optionAccountByUerId(long userId); + + //根据用户id修改 + int modify(long userId); + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/QhscGangjiaosuoService.java b/src/main/java/com/msdw/tms/service/QhscGangjiaosuoService.java new file mode 100644 index 0000000..33c60fc --- /dev/null +++ b/src/main/java/com/msdw/tms/service/QhscGangjiaosuoService.java @@ -0,0 +1,23 @@ +package com.msdw.tms.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.msdw.tms.common.utils.PageUtils; +import com.msdw.tms.entity.QhscGangjiaosuoEntity; + +import java.util.List; + +/** + * @描述: service接口 + * @作者: Rong + * @日期: 2020-10-10 + */ +public interface QhscGangjiaosuoService extends IService { + + //外汇期货:根据币种分页查询 + PageUtils queryPageByQh(Integer page, Integer size, String name, String date); + + //查询所有币种 + List getName(); + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/RedisService.java b/src/main/java/com/msdw/tms/service/RedisService.java new file mode 100644 index 0000000..d5d671d --- /dev/null +++ b/src/main/java/com/msdw/tms/service/RedisService.java @@ -0,0 +1,22 @@ +package com.msdw.tms.service; + +import com.msdw.tms.common.utils.RedisConfig; +import org.springframework.stereotype.Service; + +import javax.annotation.Resource; + +/** + * @ProjectName: tms + * @Package: com.msdw.tms.service + * @ClassName: RedisService + * @Description: java类作用描述 + * @Author: Maureen.Rong + * @CreateDate: 2020/11/13 10:21 + * @UpdateDate: 2020/11/13 10:21 + * @Version: 1.0 + */ +@Service +public class RedisService { + @Resource + private RedisConfig redisConfig; +} diff --git a/src/main/java/com/msdw/tms/service/WhscRenminbipinzhongService.java b/src/main/java/com/msdw/tms/service/WhscRenminbipinzhongService.java new file mode 100644 index 0000000..242f1ea --- /dev/null +++ b/src/main/java/com/msdw/tms/service/WhscRenminbipinzhongService.java @@ -0,0 +1,34 @@ +package com.msdw.tms.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.msdw.tms.common.utils.PageUtils; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.entity.WhscRenminbipinzhongEntity; + +import javax.servlet.http.HttpServletResponse; +import java.util.List; + +/** + * @描述:外汇市场人民币品种 service接口 + * @作者: Rong + * @日期: 2020-09-28 + */ +public interface WhscRenminbipinzhongService extends IService { + + + //导出Excel + R exportProjectRecord(HttpServletResponse response, String name, String startTime, String endTime) throws Exception; + + + //分页查询外汇人民币币种 + PageUtils queryPageByRecode(Integer pageNum, Integer pageSize, String name, String date, Integer userId); + + //根据筛选的名称查询 + R getInfobyId(String name, String startTime, String endTime, String confidenceLevel, String validity, Integer type, String experimentsNumber); + + + //查询所有汇率类型 + List getExchangeRate(); + + +} \ No newline at end of file diff --git a/src/main/java/com/msdw/tms/service/impl/AliyunOssServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/AliyunOssServiceImpl.java new file mode 100644 index 0000000..5a687f4 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/AliyunOssServiceImpl.java @@ -0,0 +1,56 @@ +package com.msdw.tms.service.impl; + +import com.aliyun.oss.OSSClient; +import com.msdw.tms.common.utils.AliyunOssUtil; +import com.msdw.tms.common.utils.FilesResult; +import com.msdw.tms.config.AliyunOssConfig; +import com.msdw.tms.service.AliyunOssService; +import org.springframework.beans.factory.BeanFactory; +import org.springframework.stereotype.Service; +import org.springframework.web.multipart.MultipartFile; + +import javax.annotation.Resource; +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; + +/** + * 阿里云OSSUtil + */ +@Service +public class AliyunOssServiceImpl implements AliyunOssService { + + @Resource + AliyunOssConfig config; + + @Resource + BeanFactory beanFactory; + + /** + * 上传文件 + * + * @param file 上传文件 + */ + public FilesResult uploadFiles(MultipartFile file) throws IOException { + OSSClient client = beanFactory.getBean(OSSClient.class); + return AliyunOssUtil.uploadFiles(file, client, config); + } + + /** + * 下载文件 + * + * @param objectName 本地路径 + */ + public void downloadFiles(HttpServletResponse response, String objectName) throws IOException { + OSSClient client = beanFactory.getBean(OSSClient.class); + AliyunOssUtil.downloadFiles(response, client, config, objectName); + } + + /** + * 根据文件路径+文件名称,删除该文件 + * + */ + public void deleteFile(String filename) { + OSSClient client = beanFactory.getBean(OSSClient.class); + AliyunOssUtil.deleteFile(client, config.getBucketName(), filename); + } +} diff --git a/src/main/java/com/msdw/tms/service/impl/CdBankStatementsServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/CdBankStatementsServiceImpl.java new file mode 100644 index 0000000..a7e3bf7 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/CdBankStatementsServiceImpl.java @@ -0,0 +1,358 @@ +package com.msdw.tms.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.trading_rules.algorithm; +import com.msdw.tms.dao.*; +import com.msdw.tms.entity.*; +import com.msdw.tms.model.req.CdGetAccountInfoReq; +import com.msdw.tms.model.resp.CdAccountInfoResp; +import com.msdw.tms.model.resp.CdTransferRecordsResp; +import com.msdw.tms.service.CdBankStatementsService; +import org.springframework.data.redis.core.RedisTemplate; +import org.springframework.stereotype.Service; + +import javax.annotation.Resource; +import java.math.BigDecimal; +import java.util.ArrayList; +import java.util.List; + +/** + * @描述:账户流水记录 服务类 + * @作者: Rong + * @日期: 2020-10-15 + */ +@Service("cdBankStatementsService") +public class CdBankStatementsServiceImpl extends ServiceImpl implements CdBankStatementsService { + + + @Resource + private CdBankStatementsDao dao; + + //期权账户 + @Resource + private CdUserOptionAccountDao accountDao; + + + //银行账户 + @Resource + private CdUserAssetsDao bankDao; + + @Resource + private QhscGangjiaosuoDao gangjiaosuoDao; + + @Resource + private CdFuturesConfigureDao configureDao; + @Resource + public RedisTemplate template; + + + /** + * 转账后添加转账流水 + * 对相应的账户做修改操作 + * + * @param + * @param + * @return + */ + @Override + //@Transactional(propagation = Propagation.NESTED, isolation = Isolation.DEFAULT, readOnly = false, rollbackFor = Exception.class) + public boolean modify(Integer userId, Integer transferOut, BigDecimal amountMoney) { + try { + + String getDate = template.opsForValue().get("date:userId_" + userId); + + //获取当前银行可用资金 + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", userId); + //getBankBalance:获取到当前银行可用资金 + BigDecimal getBankBalance = bankDao.selectOne(queryWrapper).getBalance(); + + //获取当前期权账户可用资金 + QueryWrapper queryWrapper2 = new QueryWrapper(); + queryWrapper2.eq("user_id", userId); + //getBalance:获取到当前期权账户可用资金 + BigDecimal getBalance = accountDao.selectOne(queryWrapper2).getBalance(); + + + UpdateWrapper updateWrapper1 = new UpdateWrapper(); + UpdateWrapper updateWrapper2 = new UpdateWrapper(); + CdBankStatementsEntity bankStatements1 = new CdBankStatementsEntity(); + CdBankStatementsEntity bankStatements2 = new CdBankStatementsEntity(); + + updateWrapper1.eq("user_id", userId); + updateWrapper2.eq("user_id", userId); + + if (transferOut == 1) { + /* 操作1:银行账户扣钱 期权账户加钱 同时添加流水记录*/ + + + //银行账户扣钱操作 + updateWrapper1.set("balance", getBankBalance.subtract(amountMoney)); + int updateBank = bankDao.update(null, updateWrapper1); + + + //期权账户加钱操作 + updateWrapper2.set("balance", getBalance.add(amountMoney)); + int updateAssets = accountDao.update(null, updateWrapper2); + + + //添加银行账户流水记录 + bankStatements1.setType(1);//类型(1银行账户 2期货账户) + bankStatements1.setCoinType("人民币"); + bankStatements1.setBeforeAmount(getBankBalance);//改变前金额 + bankStatements1.setAmountMoney(amountMoney);//金额 + bankStatements1.setAfterAmount(getBankBalance.subtract(amountMoney)); + bankStatements1.setTransferType(1);//流水类型(1:转账 2:流水) + bankStatements1.setRemark("转账"); + bankStatements1.setUserId(userId); + bankStatements1.setCreateTime(getDate); + int addBankStatements1 = baseMapper.insert(bankStatements1); + + //添加期权账户流水记录 + bankStatements2.setType(2);//类型(1银行账户 2期货账户) + bankStatements2.setCoinType("人民币"); + bankStatements2.setBeforeAmount(getBalance);//改变前金额 + bankStatements2.setAmountMoney(amountMoney);//金额 + bankStatements2.setAfterAmount(getBalance.add(amountMoney)); + bankStatements2.setTransferType(1);//流水类型(1:转账 2:流水) + bankStatements2.setUserId(userId); + bankStatements2.setRemark("入账"); + bankStatements2.setCreateTime(getDate); + int addBankStatements2 = baseMapper.insert(bankStatements2); + + + if (updateAssets <= 0 && updateBank <= 0 && addBankStatements1 <= 0 && addBankStatements2 <= 0) { + return false; + } + + + } else if (transferOut == 2) { + //以期权账户转出到银行账户 + /* 操作2:期权账户扣钱 银行账户加钱 同时添加流水记录*/ + + + //期权账户扣钱 + updateWrapper2.set("balance", getBalance.subtract(amountMoney)); + updateWrapper2.eq("user_id", userId); + int updateAssets = accountDao.update(null, updateWrapper2); + + + //银行账户加钱 + updateWrapper1.set("balance", getBankBalance.add(amountMoney)); + updateWrapper1.eq("user_id", userId); + int updateBank = bankDao.update(null, updateWrapper1); + + + //添加银行账户流水记录 + + bankStatements1.setType(2);//类型(1银行账户 2期货账户) + bankStatements1.setCoinType("人民币"); + bankStatements1.setBeforeAmount(getBankBalance);//改变前金额 + bankStatements1.setAmountMoney(amountMoney);//金额 + bankStatements1.setAfterAmount(getBankBalance.add(amountMoney)); + bankStatements1.setTransferType(1);//流水类型(1:转账 2:流水) + bankStatements1.setRemark("转账"); + bankStatements1.setCreateTime(getDate); + bankStatements1.setUserId(userId); + int addBankStatements1 = baseMapper.insert(bankStatements1); + + //添加期权账户流水记录 + + bankStatements2.setType(1);//类型(1银行账户 2期货账户) + bankStatements2.setCoinType("人民币"); + bankStatements2.setBeforeAmount(getBalance);//改变前金额 + bankStatements2.setAmountMoney(amountMoney);//金额 + bankStatements2.setAfterAmount(getBalance.subtract(amountMoney)); + bankStatements2.setTransferType(1);//流水类型(1:转账 2:流水) + bankStatements2.setRemark("入账"); + bankStatements2.setCreateTime(getDate); + bankStatements2.setUserId(userId); + int addBankStatements2 = baseMapper.insert(bankStatements2); + + + if (updateAssets <= 0 && updateBank <= 0 && addBankStatements1 <= 0 && addBankStatements2 <= 0) { + return false; + } + + + } + return true; + } catch (Exception e) { + e.getMessage(); + return false; + } + + + } + + /** + * 转账交易记录 + * + * @param + * @param + * @param userId + * @param startTime + * @param endTime + * @return + */ + @Override + public List queryPageByRecode(String userId, String startTime, String endTime) { + //分页对象 + QueryWrapper queryWrapper = new QueryWrapper(); + //判断条件值是否为空,如果不为空,拼接条件 + if (userId != null) { + queryWrapper.eq("user_id", userId); + } + + + if (startTime != null && endTime != null) { + if (!startTime.isEmpty() && !endTime.isEmpty()) { + queryWrapper.ge("create_time", startTime) + .le("create_time", endTime); + } + } + + //流水类型(1:转账 2:流水) + queryWrapper.eq("transfer_type", 1); + queryWrapper.eq("remark", "转账"); + + //获取list + List list = baseMapper.selectList(queryWrapper); + + //添加进新的集合 + List recordsReqs = new ArrayList<>(); + + + //类型(1银行账户 2期货账户) + for (CdBankStatementsEntity entity : list) { + CdTransferRecordsResp req = new CdTransferRecordsResp(); + if (entity.getType() == 1) { + req.setOut("银行账户"); + req.setEnter("商品期货保证金账户"); + } else if (entity.getType() == 2) { + req.setOut("商品期货保证金账户"); + req.setEnter("银行账户"); + + } + req.setCoinType("人民币"); + req.setAmountMoney(entity.getAmountMoney()); + req.setCreateTime(entity.getCreateTime()); + req.setType("自由转账"); + recordsReqs.add(req); + } + return recordsReqs; + } + + /** + * 查看期权账户时候返回头寸配置信息 + * + * @param req + * @return + */ + @Override + public R getAccountInfoReturnOpen(CdGetAccountInfoReq req, Integer userId) { + + String getDate = template.opsForValue().get("date:userId_" + userId); + + QueryWrapper account = new QueryWrapper(); + account.eq("user_id", userId); + + //查询当前账户金额 + CdUserOptionAccountEntity entity = accountDao.selectOne(account); + + CdAccountInfoResp accountInfoResp = new CdAccountInfoResp(); + //可用资金 + accountInfoResp.setBalance(entity.getBalance()); + //冻结资金 + accountInfoResp.setForzenBalance(entity.getForzenBalance()); + //占用保证金 + accountInfoResp.setDepositBalance(entity.getDepositBalance()); + + //每手交易单位(合约面值) + Integer tradingUnit = 0; + + //查询当前用户的开仓信息 + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", userId); + + BigDecimal float_pl = new BigDecimal(0); + BigDecimal marketAssets = new BigDecimal(0); + //查询当前用户持仓信息 + CdFuturesConfigureEntity configureEntity = configureDao.selectOne(queryWrapper); + if (configureEntity != null) { + + //根据pid查询当前代码 + QhscGangjiaosuoEntity qhscGangjiaosuo = gangjiaosuoDao.selectById(configureEntity.getPid()); + + //根据代码查询当天汇率价格 + QueryWrapper wrapper = new QueryWrapper(); + wrapper.eq("e_name", qhscGangjiaosuo.getEName()); + wrapper.eq("date", getDate); + QhscGangjiaosuoEntity gangjiaosuo = gangjiaosuoDao.selectOne(wrapper); + + tradingUnit = CdFuturesConfigureServiceImpl.setCompany(gangjiaosuo.getName()); + + //浮动盈亏 =(当天结算价—昨日结算价)* 委托数量 * 合约面值 + float_pl = algorithm.getFloatingPL(gangjiaosuo.getLastPrice(), gangjiaosuo.getDayUp(), new BigDecimal(tradingUnit), configureEntity.getEntrustNumber()); + accountInfoResp.setFloatingPL(float_pl); + //资产市值 市值=外汇期货合约价格 * 委托数量 * 合约面值 + + //查询持仓信息状态是开仓还是平仓(0.开仓 1.平仓) + if (configureEntity.getStatus() == 0) { + + //资产市值 = (开仓或者平仓) * 委托数量 * 合约面值(1份合约=一手,一手=?吨) + marketAssets = algorithm.getMarketValue(configureEntity.getEntrustPrice(), configureEntity.getEntrustNumber(), tradingUnit); + + } else if (configureEntity.getStatus() == 1) { + marketAssets = algorithm.getMarketValue(configureEntity.getCloseRate(), configureEntity.getEntrustNumber(), tradingUnit); + } + + accountInfoResp.setMarketAssets(marketAssets); + } else { + accountInfoResp.setMarketAssets(marketAssets); + accountInfoResp.setFloatingPL(float_pl); + } + + + //总资金 = 资产市值 + 占用保证金 + 冻结资金 + 可用资金 + BigDecimal totalAssets = marketAssets.add(accountInfoResp.getDepositBalance().add(accountInfoResp.getBalance().add(accountInfoResp.getForzenBalance()))); + accountInfoResp.setTotalAssets(totalAssets); + + + QhscGangjiaosuoEntity qh = gangjiaosuoDao.selectById(req.getId()); + + //序号 + accountInfoResp.setId(req.getId()); + //代码 + accountInfoResp.setEname(qh.getEName()); + //合约名称 + accountInfoResp.setName(qh.getName()); + //交易方向 + accountInfoResp.setTradingDirection(req.getTradingDirection()); + //需要资金 + + int unit = CdFuturesConfigureServiceImpl.setCompany(qh.getName()); + + //所需资金 = 委托价格 * 委托数量(手数) * 每手交易单位 + BigDecimal totalAmount = algorithm.getFundsRequired(req.getEntrustPrice(), BigDecimal.valueOf(req.getEntrustNumber()), unit); + accountInfoResp.setFundingNeeded(totalAmount); + + //创建时间 + accountInfoResp.setCreateTime(getDate); + //最后交易日 + //获取最后交易日 + String tradeTime = algorithm.getLastTradingDay(qh.getName()); + accountInfoResp.setTradeTime(tradeTime); + + return R.ok().put("data", accountInfoResp); + + } + + +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/CdCacheServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/CdCacheServiceImpl.java new file mode 100644 index 0000000..2b05d95 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/CdCacheServiceImpl.java @@ -0,0 +1,70 @@ +package com.msdw.tms.service.impl; + +import com.alibaba.fastjson.JSONObject; +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.dao.CdCacheDao; +import com.msdw.tms.entity.CdCacheEntity; +import com.msdw.tms.service.CdCacheService; +import org.springframework.data.redis.core.RedisTemplate; +import org.springframework.stereotype.Service; + +import javax.annotation.Resource; + +/** + * @描述:步骤缓存记录表 服务类 + * @作者: Rong + * @日期: 2020-10-21 + */ +@Service("cdCacheService") +public class CdCacheServiceImpl extends ServiceImpl implements CdCacheService { + + @Resource + private CdCacheDao dao; + + @Resource + public RedisTemplate template; + + /** + * 步骤保存 + * + * @param userId + * @param stepNumber + * @param json + * @return + */ + @Override + public R cache(Integer userId, Integer stepNumber, JSONObject json) { + //先查询用户有没有存在记录 + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("user_id", userId); + CdCacheEntity cache = baseMapper.selectOne(queryWrapper); + + System.out.println("打印:" + "stepNumber:" + stepNumber + "\tjson:"); + + if (cache == null) { + //表示第一次进入 + CdCacheEntity entity = new CdCacheEntity(); + entity.setUserId(userId); + entity.setStep1(json.toJSONString()); + entity.setStepNumber(1); + int flag = baseMapper.insert(entity); + if (flag == 1) { + return R.ok(); + } + } + //如果存在记录 则根据相应的步骤数 填入相应的Json字符串 + UpdateWrapper updateWrapper = new UpdateWrapper<>(); + updateWrapper.eq("user_id", userId); + updateWrapper.set("step_number", stepNumber); + updateWrapper.set("step" + stepNumber, json.toJSONString()); + dao.update(null, updateWrapper); + + return R.ok(); + } +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/CdExperimentalServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/CdExperimentalServiceImpl.java new file mode 100644 index 0000000..3cb4a7f --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/CdExperimentalServiceImpl.java @@ -0,0 +1,259 @@ +package com.msdw.tms.service.impl; + +import cn.afterturn.easypoi.excel.ExcelExportUtil; +import cn.afterturn.easypoi.excel.entity.ExportParams; +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.dao.CdExperimentalDao; +import com.msdw.tms.entity.CdExperimentalEntity; +import com.msdw.tms.entity.vo.CdExperimentalEntityVo; +import com.msdw.tms.service.CdExperimentalService; +import org.apache.poi.ss.usermodel.Workbook; +import org.springframework.stereotype.Service; + +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; +import java.net.URLEncoder; +import java.util.ArrayList; +import java.util.List; + +/** + * @描述:实验数据 服务类 + * @作者: Rong + * @日期: 2020-10-20 + */ +@Service("cdExperimentalService") +public class CdExperimentalServiceImpl extends ServiceImpl implements CdExperimentalService { + + /** + * 单条导出实验结果 + * + * @param response + * @param id + * @return + * @throws Exception + */ + @Override + public R exportProjectRecord(HttpServletResponse response, Integer id) throws Exception { + + try { + + CdExperimentalEntity entity = baseMapper.selectById(id); + if (entity == null) { + return R.ok("暂无此数据"); + } + + QueryWrapper wrapper = new QueryWrapper(); + wrapper.eq("id", id); + List list = baseMapper.selectList(wrapper); + switch (entity.getType()) { + + case 0: + case 1: + + if (list != null) { + + // 告诉浏览器用什么软件可以打开此文件 + response.setHeader("content-Type", "application/vnd.ms-excel"); + // 下载文件的默认名称 + response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("实验数据", "UTF-8") + ".xls"); + //编码 + response.setCharacterEncoding("UTF-8"); + Workbook workbook = ExcelExportUtil.exportExcel(new ExportParams(), CdExperimentalEntity.class, list); + workbook.write(response.getOutputStream()); + } + + break; + case 2: + + + CdExperimentalEntity entity1 = baseMapper.selectById(id); + CdExperimentalEntityVo vo = new CdExperimentalEntityVo(); + + vo.setId(entity1.getId()); + vo.setExchangeRate(entity1.getExchangeRate()); + vo.setSampleData(entity1.getSampleData()); + vo.setConfidenceLevel(entity1.getConfidenceLevel()); + vo.setValidPeriod(entity1.getValidPeriod()); + vo.setTrendChart(entity1.getTrendChart()); + vo.setProfitLossDistribution(entity1.getProfitLossDistribution()); + vo.setCreatTime(entity1.getCreatTime()); + + List voList = new ArrayList<>(); + voList.add(vo); + + + // 告诉浏览器用什么软件可以打开此文件 + response.setHeader("content-Type", "application/vnd.ms-excel"); + // 下载文件的默认名称 + response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("实验数据(蒙特卡洛)", "UTF-8") + ".xls"); + //编码 + response.setCharacterEncoding("UTF-8"); + Workbook workbook = ExcelExportUtil.exportExcel(new ExportParams(), CdExperimentalEntityVo.class, voList); + workbook.write(response.getOutputStream()); + + + break; + + } + + + } catch (IOException e) { + e.printStackTrace(); + return R.error("出现异常:" + e.getMessage()); + + } + return R.ok(); + } + + /** + * 根据选择的类型导出实验结果 + * + * @param response + * @param userId + * @return + * @throws Exception 要导出的类型-实验类型(0正态Var,1历史模拟,2蒙特卡洛)" + */ + @Override + public R exportexportProjectRecordByAll(HttpServletResponse response, Integer userId, Integer type) throws Exception { + + //查询该用户做过的实验 + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", userId); + queryWrapper.eq("type", type); + List list = baseMapper.selectList(queryWrapper); + + if (list == null) { + return R.ok("没有该数据,无法导出Excel。"); + } + List voList = new ArrayList<>(); + switch (type) { + + case 0: + case 1: + + + // 告诉浏览器用什么软件可以打开此文件 + response.setHeader("content-Type", "application/vnd.ms-excel"); + // 下载文件的默认名称 + response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("实验数据", "UTF-8") + ".xls"); + //编码 + response.setCharacterEncoding("UTF-8"); + Workbook workbook = ExcelExportUtil.exportExcel(new ExportParams(), CdExperimentalEntity.class, list); + workbook.write(response.getOutputStream()); + + + break; + case 2: + + + for (CdExperimentalEntity experimentalEntity : list) { + CdExperimentalEntityVo vo = new CdExperimentalEntityVo(); + System.out.println("打印:>>>>>>>" + experimentalEntity.getId()); + vo.setId(experimentalEntity.getId()); + vo.setExchangeRate(experimentalEntity.getExchangeRate()); + vo.setSampleData(experimentalEntity.getSampleData()); + vo.setConfidenceLevel(experimentalEntity.getConfidenceLevel()); + vo.setValidPeriod(experimentalEntity.getValidPeriod()); + vo.setTrendChart(experimentalEntity.getTrendChart()); + vo.setProfitLossDistribution(experimentalEntity.getProfitLossDistribution()); + vo.setCreatTime(experimentalEntity.getCreatTime()); + + System.out.println("赋值前打印:" + vo); + + voList.add(vo); + } + + + // 告诉浏览器用什么软件可以打开此文件 + response.setHeader("content-Type", "application/vnd.ms-excel"); + // 下载文件的默认名称 + response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("实验数据(蒙特卡洛)", "UTF-8") + ".xls"); + //编码 + response.setCharacterEncoding("UTF-8"); + Workbook workbook1 = ExcelExportUtil.exportExcel(new ExportParams(), CdExperimentalEntityVo.class, voList); + workbook1.write(response.getOutputStream()); + break; + + } + + return null; + + } + + /** + * 根据选择的id导出实验 + * + * @param response + * @param userId + * @param ids + * @return + * @throws Exception + */ + @Override + public R exportexportProjectRecordById(HttpServletResponse response, Integer userId, List ids) throws Exception { + + List list = this.listByIds(ids); + if (list == null) { + return R.ok("没有该数据,无法导出Excel。"); + } + List voList = new ArrayList<>(); + for (CdExperimentalEntity experimental : list) { + switch (experimental.getType()) { + case 0: + case 1: + + + // 告诉浏览器用什么软件可以打开此文件 + response.setHeader("content-Type", "application/vnd.ms-excel"); + // 下载文件的默认名称 + response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("实验数据", "UTF-8") + ".xls"); + //编码 + response.setCharacterEncoding("UTF-8"); + Workbook workbook = ExcelExportUtil.exportExcel(new ExportParams(), CdExperimentalEntity.class, list); + workbook.write(response.getOutputStream()); + + + break; + case 2: + + for (CdExperimentalEntity experimentalEntity : list) { + CdExperimentalEntityVo vo = new CdExperimentalEntityVo(); + System.out.println("打印:>>>>>>>" + experimentalEntity.getId()); + vo.setId(experimentalEntity.getId()); + vo.setExchangeRate(experimentalEntity.getExchangeRate()); + vo.setSampleData(experimentalEntity.getSampleData()); + vo.setConfidenceLevel(experimentalEntity.getConfidenceLevel()); + vo.setValidPeriod(experimentalEntity.getValidPeriod()); + vo.setTrendChart(experimentalEntity.getTrendChart()); + vo.setProfitLossDistribution(experimentalEntity.getProfitLossDistribution()); + vo.setCreatTime(experimentalEntity.getCreatTime()); + + System.out.println("赋值前打印:" + vo); + + voList.add(vo); + } + + + // 告诉浏览器用什么软件可以打开此文件 + response.setHeader("content-Type", "application/vnd.ms-excel"); + // 下载文件的默认名称 + response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("实验数据(蒙特卡洛)", "UTF-8") + ".xls"); + //编码 + response.setCharacterEncoding("UTF-8"); + Workbook workbook1 = ExcelExportUtil.exportExcel(new ExportParams(), CdExperimentalEntityVo.class, voList); + workbook1.write(response.getOutputStream()); + break; + } + } + return null; + + + } + + +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/CdFuturesConfigureServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/CdFuturesConfigureServiceImpl.java new file mode 100644 index 0000000..52169c3 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/CdFuturesConfigureServiceImpl.java @@ -0,0 +1,964 @@ +package com.msdw.tms.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.common.utils.MoneyConversionUtil; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.date.DateUtils; +import com.msdw.tms.common.utils.py.PyUtil; +import com.msdw.tms.common.utils.trading_rules.algorithm; +import com.msdw.tms.dao.*; +import com.msdw.tms.entity.*; +import com.msdw.tms.entity.vo.CdFuturesConfigureEntityVo; +import com.msdw.tms.model.resp.CdAfterHedgingResp; +import com.msdw.tms.model.resp.CdBeforeHedgingResp; +import com.msdw.tms.model.resp.CdHoldPositionInfoResp; +import com.msdw.tms.model.resp.CdTransferWarehouseDetailResp; +import com.msdw.tms.service.CdFuturesConfigureService; +import org.springframework.data.redis.core.RedisTemplate; +import org.springframework.stereotype.Service; + +import javax.annotation.Resource; +import java.io.IOException; +import java.math.BigDecimal; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +/** + * @描述:期货资产头寸配置 服务类 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Service("cdFuturesConfigureService") +public class CdFuturesConfigureServiceImpl extends ServiceImpl implements CdFuturesConfigureService { + + + @Resource + public QhscGangjiaosuoDao qhscGangjiaosuoDao; + + @Resource + public CdFuturesConfigureDao futuresConfigureDao; + + @Resource + public CdUserOptionAccountDao accountDao; + + @Resource + public CdBankStatementsDao bankStatementsDao; + + @Resource + public CdTransferWarehouseRecordDao transferWarehouseRecordDao; + + @Resource + public RedisTemplate template; + @Resource + public WhscRenminbipinzhongDao renminbipinzhongDao; + + //开仓处理 + public R openPosition(CdFuturesConfigureEntity entity) { + try { + + //判断返回上一步重新开仓的情况下 + + //1.查询该用户有没有建仓信息 + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", entity.getUserId()); + CdFuturesConfigureEntity check = baseMapper.selectOne(queryWrapper); + + //check不为空的情况下说明有开仓信息 再看有没有修改参数 + // 修改了就删除当前的开仓信息 + if (check != null) { + //先查看流水表要返回的金额 + QueryWrapper wrapper = new QueryWrapper<>(); + wrapper.eq("user_id", entity.getUserId()); + wrapper.eq("transfer_type", "2"); + wrapper.eq("type", "2").last("limit 0,1"); + CdBankStatementsEntity get = bankStatementsDao.selectOne(wrapper); + + //获取到第一条改变前的金额 + if (get != null) { + //将金额回滚到改变前的金额 + + //做修改账户金额操作 + UpdateWrapper updateWrapper2 = new UpdateWrapper(); + updateWrapper2.set("balance", get.getBeforeAmount()); + //修改占用保证金 getPreFreezingMargin + updateWrapper2.set("deposit_balance", 0); + updateWrapper2.set("forzen_balance", 0); + updateWrapper2.eq("user_id", entity.getUserId()); + int updateAssets = accountDao.update(null, updateWrapper2); + + //删除(开仓、平仓、调仓)操作流水表 + + QueryWrapper delRecord = new QueryWrapper<>(); + delRecord.eq("cid", check.getId()); + int delRecodeByOpen = transferWarehouseRecordDao.delete(delRecord); + + //删除头寸表记录 + int delOpen = baseMapper.deleteById(check.getId()); + + + //删除账户流水表记录 + Map map = new HashMap<>(); + //删除类型为流水的 : (1:转账 2:流水) + map.put("transfer_type", "2"); + map.put("user_id", entity.getUserId()); + + int delAccount = bankStatementsDao.deleteByMap(map); + + if (updateAssets <= 0 && delRecodeByOpen <= 0 && delOpen <= 0 && delAccount <= 0) { + return R.error(); + } + } + + + } + + + String getDate = template.opsForValue().get("date:userId_" + entity.getUserId()); + //python表:根据合约id查询合约相关数据 + QhscGangjiaosuoEntity qh = qhscGangjiaosuoDao.selectById(entity.getPid()); + + if (qh == null) { + return R.error("暂时没有该期货合约的最新价"); + } + + //委托价格,委托数量为空默认为合约当前最新价格,委托数量为1手 + if (entity.getEntrustPrice() == null && entity.getEntrustNumber() == null) { + //设置委托价格 + entity.setEntrustPrice(qh.getLastPrice()); + //设置委托数量 + entity.setEntrustNumber(1); + + } else if (entity.getEntrustPrice() != null && entity.getEntrustNumber() == null) { + //委托数量不为空时,委托数量为null 默认1手 + + //设置委托价格 + entity.setEntrustPrice(entity.getEntrustPrice()); + //设置委托数量 + entity.setEntrustNumber(1); + } else { + //根据前端传的值设置 + entity.setEntrustPrice(entity.getEntrustPrice()); + entity.setEntrustNumber(entity.getEntrustNumber()); + + } + //设置最新价格 + entity.setNewestPrice(qh.getLastPrice()); + + //委托价 + double entrustPrice = entity.getEntrustPrice().doubleValue(); + + //最新价 + double newestPrice = entity.getNewestPrice().doubleValue(); + + /** + * 期货的开仓成交规则: + * 多方情况下,委托价格大于等于最新价,以最新价成交; + * 空方情况下,委托价格小于等于最新价,以最新价成交。 + */ + //交易方向(0空头,1多头) + switch (entity.getTradingDirection()) { + + case 0: + //空方情况下:委托价格小于等于最新价,以最新价成交 + // entrustPrice > newestPrice + if (entrustPrice > newestPrice) { + return R.error("开仓失败,委托价格大于最新价"); + } + break; + case 1: + //多方情况下:委托价格大于等于最新价,以最新价成交 + // entrustPrice < newestPrice + if (entrustPrice < newestPrice) { + return R.error("开仓失败,委托价格小于最新价"); + } + break; + + } + + //每手交易单位(合约面值) + Integer tradingUnit = 0; + + + tradingUnit = setCompany(qh.getName()); + + //设置最新价格 + //entity.setNewestPrice(qh.getLastPrice()); + + + //所需资金 = 委托价格 * 每手交易单位 * 委托数量(手数) + BigDecimal totalAmount = algorithm.getFundsRequired(entity.getEntrustPrice(), BigDecimal.valueOf(entity.getEntrustNumber()), tradingUnit); + + + // + + //所需资金:与前端传来参数对比 需要资金 = 委托价格 * 每手交易单位 * 委托数量(手数) + if (entity.getFundingNeeded().compareTo(totalAmount) == 0) { + //a.compareTo(b) == 0 + entity.setFundingNeeded(totalAmount); + + //浮动盈亏=(当天结算价—昨日结算价)* 委托数量 * 合约面值 + BigDecimal getFloatingPL = algorithm.getFloatingPL(qh.getLastPrice(), qh.getDayUp(), new BigDecimal(entity.getEntrustNumber()), tradingUnit); + entity.setFloatingPl(getFloatingPL); + + //获取最后交易日 + String tradeTime = algorithm.getLastTradingDay(qh.getName()); + entity.setTradeTime(tradeTime); + + //pre_freezing_margin:预冻结保证金 = 外汇期货合约价格 * 合约面值 * 手数 * 保证金比例 + BigDecimal getPreFreezingMargin = algorithm.getPreFreezingMargin(qh.getLastPrice(), new BigDecimal(tradingUnit), qh.getName().substring(0, 8)); + entity.setPreFreezingMargin(getPreFreezingMargin); + + //pre_freezing_frozen:预冻结手续费 = 委托数量(手)* 费用标准 + BigDecimal getPreFreezingServiceCharge = algorithm.getPreFreezingServiceCharge(new BigDecimal(entity.getEntrustNumber())); + entity.setPreFreezingFrozen(getPreFreezingServiceCharge); + + + //frozen_price:冻结资金 = 预计冻结保证金+预冻结手续费 + BigDecimal frozen_price = getPreFreezingMargin.add(getPreFreezingServiceCharge); + entity.setFrozenPrice(frozen_price); + + + //获取当前期权账户可用资金 + QueryWrapper queryWrapper2 = new QueryWrapper(); + queryWrapper2.eq("user_id", entity.getUserId()); + //getBalance:获取到当前期权账户可用资金 + BigDecimal nowBalance = accountDao.selectOne(queryWrapper2).getBalance(); + + + //deduction:要扣除的金额 = 冻结资金 + 所需资金 + BigDecimal deduction = frozen_price.add(totalAmount); + + //modifyPrice:要修改的金额(需要扣除的金额 = 当前账户可用资金 - 要扣除的金额) + BigDecimal modifyPrice = nowBalance.subtract(deduction); + + //判断当前可用余额是否足以扣除本次交易金额 >= + double nowBalance1 = nowBalance.doubleValue(); + double deduction1 = deduction.doubleValue(); + if (nowBalance1 < deduction1) { + return R.error("可用资金不足,开仓失败"); + } + + //设置可交易数量 = 当前的委托数量 + entity.setTradableNumber(new BigDecimal(entity.getEntrustNumber())); + + //做修改账户金额操作 + UpdateWrapper updateWrapper2 = new UpdateWrapper(); + updateWrapper2.set("balance", modifyPrice); + //修改占用保证金 getPreFreezingMargin + updateWrapper2.set("deposit_balance", getPreFreezingMargin); + updateWrapper2.eq("user_id", entity.getUserId()); + int updateAssets = accountDao.update(null, updateWrapper2); + + + //添加期权账户流水记录 + CdBankStatementsEntity bankStatements2 = new CdBankStatementsEntity(); + bankStatements2.setType(2);//类型(1银行账户 2期货账户) + bankStatements2.setCoinType("人民币"); + bankStatements2.setBeforeAmount(nowBalance);//改变前金额 + bankStatements2.setAmountMoney(deduction);//这次操作的金额 + bankStatements2.setAfterAmount(modifyPrice);//改变后的金额 + bankStatements2.setTransferType(2);//流水类型(1:转账 2:流水) + bankStatements2.setCreateTime(getDate); + bankStatements2.setUserId(entity.getUserId()); + bankStatements2.setRemark("开仓"); + int addBankStatements2 = bankStatementsDao.insert(bankStatements2); + + + if (updateAssets <= 0 && addBankStatements2 <= 0) { + return R.error("开仓失败"); + } + + + //设置当前时间 + entity.setCreateTime(getDate); + //设置状态(0开仓,1平仓) + entity.setStatus(0); + //是否自动平仓(0否,1是) + entity.setType(0); + this.save(entity); + + //往调仓记录表添加开仓记录 + CdTransferWarehouseRecordEntity record = new CdTransferWarehouseRecordEntity(); + //开仓表id + record.setCid(entity.getId()); + record.setUserId(entity.getUserId()); + record.setDirection(0);//买卖方向(0开仓 1平仓) + record.setType(0);//委托类型(0市价 1限价) + record.setNumber(entity.getTradableNumber());//委托手数 + + + //getMarketValue:市值 = 外汇期货合约价格 * 委托数量 * 合约面值 + //BigDecimal getMarketValue = algorithm.getMarketValue(getNowPrice, configure.getEntrustNumber(), number); + + record.setBalance(deduction);//市值>>>金额 + record.setCreateTime(getDate); + transferWarehouseRecordDao.insert(record); + + + return R.ok(); + } + return R.error("所需资金对不上"); + + + } catch (Exception e) { + e.printStackTrace(); + return R.error(e.getMessage()); + } + } + + + //外汇期货持仓情况 + public CdHoldPositionInfoResp myPosition(int userId) { + + try { + QueryWrapper wrapper = new QueryWrapper<>(); + wrapper.eq("user_id", userId); + CdFuturesConfigureEntity entity = this.getOne(wrapper); + + + //查询合约表信息 + QhscGangjiaosuoEntity qh = qhscGangjiaosuoDao.selectById(entity.getPid()); + entity.setEname(qh.getEName()); + entity.setName(qh.getName()); + + CdHoldPositionInfoResp resp = new CdHoldPositionInfoResp(); + resp.setId(entity.getId()); + resp.setUserId(entity.getUserId()); + //交易方向(0空头 卖,1多头 买) + if (entity.getTradingDirection() == 0) { + resp.setTradingDirection("卖"); + } else if (entity.getTradingDirection() == 1) { + resp.setTradingDirection("买"); + } + + resp.setEntrustPrice(entity.getEntrustPrice()); + resp.setEntrustNumber(entity.getEntrustNumber()); + //持仓状态(0.开仓 1.平仓) + if (entity.getStatus() == 0) { + resp.setStatus("开仓"); + } else if (entity.getStatus() == 1) { + resp.setStatus("平仓"); + } + + resp.setCreateTime(entity.getCreateTime().substring(0, 10)); + resp.setUpdateTime(entity.getUpdateTime()); + resp.setTradeTime(entity.getTradeTime()); + resp.setName(entity.getName()); + resp.setEname(entity.getEname()); + resp.setPreFreezingMargin(entity.getPreFreezingMargin()); + + + //合约面值 + Integer tradingUnit = setCompany(entity.getName()); + + //BigDecimal todayPrice, BigDecimal yesterdayPrice, BigDecimal commissionQuantity, Integer contractFaceValue + //浮动盈亏=(当天结算价—昨日结算价)* 委托数量 * 合约面值(1份合约=一手,一手=?吨) + BigDecimal getFloatingPL = algorithm.getFloatingPL(qh.getLastPrice(), qh.getDayUp(), BigDecimal.valueOf(tradingUnit), entity.getEntrustNumber()); + resp.setFloatingPl(getFloatingPL); + + + return resp; + } catch (Exception e) { + e.printStackTrace(); + return null; + } + + } + + + /** + * 平仓操作 + * 需求:平仓前判断行情滚动时间是不等于最后交易日,行情滚动是最后交易日当天才能滚动 + * 平仓,选择当天平台的日期,已当天的行情价格,卖出或者买入相同的期货合约平仓。 + * 外汇期货盈亏=(平仓时的外汇汇率价格-下单时外汇汇率价格) * 委托数量 * 合约面值。 + * + * @param id + * @return + */ + @Override + public R buyIn(int id, String date, Integer userId) { + try { + + String getDate = template.opsForValue().get("date:userId_" + userId); + //根据id查询开仓后的信息 + CdFuturesConfigureEntity configure = baseMapper.selectById(id); + + //判断传的行情滚动日期是否等于最后交易日 + /*if (!date.trim().equals(configure.getTradeTime().substring(0, 10))) { + return R.error("行情滚动时间要与最后交易日一致"); + }*/ + + + //根据合约id表查询合约信息 + QhscGangjiaosuoEntity qh = qhscGangjiaosuoDao.selectById(configure.getPid()); + //获取该合约的代码 + String ename = qh.getEName(); + //获取每一手的单位 + Integer number = setCompany(qh.getName()); + + //根据合约代码查询平仓最后交易日那天的的汇率信息 + QueryWrapper queryWrapper2 = new QueryWrapper(); + queryWrapper2.eq("e_name", ename); + queryWrapper2.eq("date", date); + + QhscGangjiaosuoEntity now = qhscGangjiaosuoDao.selectOne(queryWrapper2); + if (now == null) { + //查询不到合约表当天的信息时候弹出 + return R.error("不在交易时段内,无法进行交易"); + } + + //查询合约当天外汇汇率价格 + double getNowPrice = now.getLastPrice().doubleValue(); + + //查询开仓时候的委托价格 + double entrustPrice = configure.getEntrustPrice().doubleValue(); + + /** + * 期货的平仓成交规则: + * configure.getTradingDirection() : 交易方向(0空头,1多头) + * 空方情况下,可交易数量为空方合约持仓总量,委托价格大于等于最新价,以最新价成交。 + * 多方情况下,可交易数量为多方合约持仓总量,委托价格小于等于最新价,以最新价成交; + */ + //TODO 暂时注释 + /* switch (configure.getTradingDirection()) { + + case 0: + //空方情况下:委托价格大于等于最新价,以最新价成交。 + if (entrustPrice < getNowPrice) { + return R.error("平仓失败,委托价格小于最新价"); + } + break; + case 1: + //多方情况下:委托价格小于等于最新价,以最新价成交; + if (entrustPrice > getNowPrice) { + return R.error("平仓失败,委托价格大于最新价"); + } + break; + + }*/ + + + //外汇期货盈亏=(平仓时的外汇汇率价格(等同于当天合约的最新价)-下单时外汇汇率价格(开仓时候的委托价)) * 委托数量 * 合约面值。 + BigDecimal pl = algorithm.get_wh_pl(BigDecimal.valueOf(getNowPrice), configure.getEntrustPrice(), new BigDecimal(configure.getEntrustNumber()), new BigDecimal(number)); + + //getMarketValue:市值 = 外汇期货合约价格 * 委托数量 * 合约面值 + BigDecimal getMarketValue = algorithm.getMarketValue(BigDecimal.valueOf(getNowPrice), configure.getEntrustNumber(), number); + + CdFuturesConfigureEntity entity = new CdFuturesConfigureEntity(); + //设置平仓时间 + entity.setUpdateTime(date); + //设置状态(0.开仓 1平仓) + entity.setStatus(1); + //设置平仓价格:市场当前价 + entity.setCloseRate(BigDecimal.valueOf(getNowPrice)); + //设置盈亏 + entity.setFloatingPl(pl); + entity.setId(id); + + //平仓时 修改冻结资金为0 + BigDecimal frozenPrice = configure.getFrozenPrice(); + entity.setFrozenPrice(new BigDecimal(0)); + + int update = baseMapper.updateById(entity); + + + //获取当前期权账户可用资金 + QueryWrapper queryWrapper3 = new QueryWrapper(); + queryWrapper3.eq("user_id", configure.getUserId()); + //nowBalance:获取到当前期权账户可用资金 + BigDecimal nowBalance = accountDao.selectOne(queryWrapper3).getBalance(); + + //改变后的金额 再加上冻结的资金 + BigDecimal newPrice = nowBalance.add(getMarketValue).add(frozenPrice); + + + //做修改账户金额操作 + UpdateWrapper modify = new UpdateWrapper(); + //平仓后加金额 = 当前可用余额 + 市值 + modify.set("balance", newPrice); + modify.eq("user_id", configure.getUserId()); + int updateAssets = accountDao.update(null, modify); + + + //添加期权账户流水记录 + CdBankStatementsEntity bankStatements2 = new CdBankStatementsEntity(); + bankStatements2.setType(2);//类型(1银行账户 2期货账户) + bankStatements2.setCoinType("人民币"); + bankStatements2.setBeforeAmount(nowBalance);//改变前金额 + bankStatements2.setAmountMoney(getMarketValue);//这次操作的金额 + bankStatements2.setAfterAmount(newPrice);//改变后的金额 + bankStatements2.setTransferType(2);//流水类型(1:转账 2:流水) + bankStatements2.setCreateTime(configure.getTradeTime()); + bankStatements2.setRemark("平仓"); + bankStatements2.setUserId(configure.getUserId()); + int addBankStatements2 = bankStatementsDao.insert(bankStatements2); + + + //往调仓记录表添加平仓记录 + CdTransferWarehouseRecordEntity record = new CdTransferWarehouseRecordEntity(); + //开仓表id + record.setCid(configure.getId()); + record.setUserId(configure.getUserId()); + record.setDirection(1);//买卖方向(0开仓 1平仓) + record.setType(0);//委托类型(0市价 1限价) + record.setNumber(configure.getTradableNumber());//委托手数 + record.setBalance(getMarketValue);//市值>>>金额 + record.setCreateTime(configure.getTradeTime()); + int addTransferWarehouseRecor = transferWarehouseRecordDao.insert(record); + + + if (addTransferWarehouseRecor <= 0 && updateAssets <= 0 && addBankStatements2 <= 0 && update <= 0) { + return R.error("平仓失败"); + } + return R.ok(); + } catch (Exception e) { + e.printStackTrace(); + return R.error(e.getMessage()); + } + } + + + //调仓前展示详情 + /*@Override + public CdTransferWarehouseDetailResp transferWarehouseByDetail(int id) { + + CdFuturesConfigureEntity futuresConfigure = baseMapper.selectById(id); + + //python表:根据合约id查询合约相关数据 + QhscGangjiaosuoEntity qh = qhscGangjiaosuoDao.selectById(futuresConfigure.getPid()); + futuresConfigure.setName(qh.getName()); + futuresConfigure.setEname(qh.getEName()); + futuresConfigure.setLastPrice(qh.getLastPrice()); + futuresConfigure.setU(qh.getU()); + futuresConfigure.setDayUp(qh.getDayUp()); + + + String have = qh.getHava(); + BigDecimal getHave = new BigDecimal(0); + if (have.contains("万")) { + have = have.substring(0, have.length() - 1); + double getHaveByDouble = Double.valueOf(have); + + getHave = MoneyConversionUtil.moneyConversion(getHaveByDouble); + + } + //用户选择的买卖方向为开仓时候:查python的持仓总量 + futuresConfigure.setTradableQuantityByBuy(getHave); + //选择为平仓时候:查自己的持仓总量 + futuresConfigure.setTradableQuantityBySell(futuresConfigure.getTradableNumber()); + + + //获取当前期权账户可用资金 + QueryWrapper queryWrapper3 = new QueryWrapper(); + queryWrapper3.eq("user_id", futuresConfigure.getUserId()); + //nowBalance:获取到当前期权账户可用资金 + BigDecimal nowBalance = accountDao.selectOne(queryWrapper3).getBalance(); + futuresConfigure.setAvailableFunds(nowBalance); + return futuresConfigure; + } +*/ + + public static void main(String[] args) { + String str = "2020-10-30 00:00:00"; + System.out.println(str.substring(0, 10)); + } + + //调仓前展示详情 + @Override + public CdTransferWarehouseDetailResp transferWarehouseByDetail(int id) { + try { + CdFuturesConfigureEntity configure = baseMapper.selectById(id); + CdTransferWarehouseDetailResp resp = new CdTransferWarehouseDetailResp(); + //python表:根据合约id查询合约相关数据 + QhscGangjiaosuoEntity qh = qhscGangjiaosuoDao.selectById(configure.getPid()); + + String have = qh.getHava(); + BigDecimal getHave = new BigDecimal(0); + if (have.contains("万")) { + have = have.substring(0, have.length() - 1); + double getHaveByDouble = Double.valueOf(have); + + getHave = MoneyConversionUtil.moneyConversion(getHaveByDouble); + + } + getHave = new BigDecimal(have); + + + //获取当前期权账户可用资金 + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", configure.getUserId()); + //nowBalance:获取到当前期权账户可用资金 + BigDecimal nowBalance = accountDao.selectOne(queryWrapper).getBalance(); + + + //id + resp.setId(configure.getId()); + //交易方向 + resp.setTradingDirection(configure.getTradingDirection()); + //合约主键 + resp.setPid(configure.getPid()); + //开仓委托价 + resp.setNewestPrice(configure.getNewestPrice()); + //名称 + resp.setName(qh.getName()); + //代码 + resp.setEname(configure.getEname()); + //市价(最新价) + resp.setLastPrice(qh.getLastPrice()); + //涨跌幅 + resp.setU(qh.getU()); + //昨结 + resp.setDayUp(qh.getDayUp()); + //买卖方向为开仓时候:可交易数量为Python表的持仓总量 + resp.setTradableQuantityByBuy(getHave); + //买卖方向为平仓时候:可交易数量为自己的持仓总量 + resp.setTradableQuantityBySell(configure.getTradableNumber()); + //当前期权账户可用资金 + resp.setAvailableFunds(nowBalance); + //买入价 + resp.setBuyPrice(qh.getBuyPrice()); + //卖出价 + resp.setCellPrice(qh.getCellPrice()); + //买量 + resp.setBuy(qh.getBuy()); + //卖量 + resp.setCell(qh.getCell()); + //总量 + resp.setSuoyou(qh.getSuoyou()); + //代码 + resp.setEname(qh.getEName()); + //涨跌额 + resp.setUdPrice(qh.getUdPrice()); + + + return resp; + } catch (NumberFormatException e) { + e.printStackTrace(); + return null; + } + } + + //外汇期货平仓绩效 + @Override + public CdFuturesConfigureEntityVo buyInAchievements(int id, Integer userId) { + String getDate = template.opsForValue().get("date:userId_" + userId); + Integer getPid = baseMapper.selectById(id).getPid(); + + //每手交易单位 + Integer tradingUnit = 0; + //python表:根据合约id查询合约相关数据 + QhscGangjiaosuoEntity qh = qhscGangjiaosuoDao.selectById(getPid); + + tradingUnit = setCompany(qh.getName()); + + + //期货开仓平仓记录 + CdFuturesConfigureEntity entity = baseMapper.selectById(id); + + + //开仓后资产市值:开仓价格 * 委托数量 * 合约面值 + BigDecimal marketValueByOpenPosition = algorithm.getMarketValue(entity.getEntrustPrice(), entity.getEntrustNumber(), tradingUnit); + + //平仓后资产市值:平仓价格 * 委托数量 * 合约面值 + BigDecimal marketValueByClosePosition = algorithm.getMarketValue(entity.getCloseRate(), entity.getEntrustNumber(), tradingUnit); + + + CdFuturesConfigureEntityVo vo = new CdFuturesConfigureEntityVo(); + vo.setId(entity.getId()); + //代码 + vo.setEName(qh.getEName()); + vo.setName(qh.getName()); + + if (entity.getTradingDirection() == 0) { + vo.setTradingDirection("做空"); + } else if (entity.getTradingDirection() == 1) { + vo.setTradingDirection("做多"); + } + + + //资产市值1 + vo.setMarketValue1(marketValueByOpenPosition); + //资产市值2 + vo.setMarketValue2(marketValueByClosePosition); + + //开仓时间 + vo.setCreateTime(DateUtils.dateForByStr(entity.getCreateTime())); + + //平仓时间 + vo.setUpdateTime(DateUtils.dateForByStr(entity.getUpdateTime())); + + //数量 + vo.setEntrustNumber(entity.getEntrustNumber()); + + //外汇盈亏=(平仓时的外汇汇率价格 - 下单时外汇汇率价格) * 委托数量 * 合约面值。 + BigDecimal pl = algorithm.get_wh_pl(entity.getCloseRate(), entity.getNewestPrice(), BigDecimal.valueOf(entity.getEntrustNumber()), BigDecimal.valueOf(tradingUnit)); + + vo.setWh_pl(pl); + vo.setCloseRate(entity.getCloseRate()); + vo.setEntrustPrice(entity.getEntrustPrice()); + + + return vo; + } + + + //外汇现货盈亏 + @Override + public CdBeforeHedgingResp beforeHedging(int userId) { + + //根据用户id查询当前的pid 以及 开仓平仓时间 + QueryWrapper wrapper = new QueryWrapper(); + wrapper.eq("user_id", userId); + CdFuturesConfigureEntity configureEntity = baseMapper.selectOne(wrapper); + + + //根据pid查询合约名称 + QhscGangjiaosuoEntity qh = qhscGangjiaosuoDao.selectById(configureEntity.getPid()); + + + //合约名称 + String name = qh.getName().substring(0, 2); + + + name += "人民币中间价"; + + //获取代码 + QueryWrapper wrapper1 = new QueryWrapper(); + wrapper1.like("name", name).last("limit 0,1"); + WhscRenminbipinzhongEntity whsc = renminbipinzhongDao.selectOne(wrapper1); + + + //每手交易单位 + Integer tradingUnit = setCompany(qh.getName().substring(0, 2) + "兑人民币期货"); + + CdBeforeHedgingResp resp = new CdBeforeHedgingResp(); + + //开仓交易时间 + String start = DateUtils.dateForByStr(configureEntity.getCreateTime()); + //平仓交易时间 + String end = DateUtils.dateForByStr(configureEntity.getUpdateTime()); + + //获取开仓时间当天的现货价格 + QueryWrapper getDateWapper = new QueryWrapper(); + getDateWapper.eq("name", name); + getDateWapper.eq("date", start); + WhscRenminbipinzhongEntity getDate1 = renminbipinzhongDao.selectOne(getDateWapper); + + //获取平仓时间当天的现货价格 + QueryWrapper getDateWapper1 = new QueryWrapper(); + getDateWapper1.eq("name", name); + getDateWapper1.eq("date", end); + WhscRenminbipinzhongEntity getDate2 = renminbipinzhongDao.selectOne(getDateWapper1); + + + //开仓金额(外币) = 汇率 * 合约面值 + BigDecimal foreignCurrencyByOpen = algorithm.getForeignCurrency(BigDecimal.valueOf(getDate1.getLastPrice()), new BigDecimal(10000000)); + + //金额(外币) = 汇率价格 * 合约面值 + BigDecimal foreignCurrencyByClose = algorithm.getForeignCurrency(BigDecimal.valueOf(getDate2.getLastPrice()), new BigDecimal(10000000)); + + + //盈亏 + BigDecimal profitLoss = new BigDecimal(0).setScale(2, BigDecimal.ROUND_HALF_UP); + + profitLoss = foreignCurrencyByOpen.subtract(foreignCurrencyByClose); + + + //交易方向(0空头 卖,1多头 买) + if (configureEntity.getTradingDirection() == 0) { + resp.setTradingDirection("卖"); + } else if (configureEntity.getTradingDirection() == 1) { + resp.setTradingDirection("买"); + } + resp.setProfitLoss(profitLoss); + resp.setName(qh.getName().substring(0, 6)); + + + //金额外币(开仓) + resp.setForeignCurrencyByOpen(new BigDecimal(10000000)); + //金额外币(平仓) + resp.setForeignCurrencyByClose(new BigDecimal(10000000)); +/* + //开仓所需资金 + BigDecimal openNeed = configureEntity.getEntrustPrice().multiply(foreignCurrencyByOpen); + //平仓所需资金 + BigDecimal closeNeed = configureEntity.getCloseRate().multiply(foreignCurrencyByClose);*/ + + //开仓所需资金 + BigDecimal openNeed = new BigDecimal(getDate1.getLastPrice()).multiply(foreignCurrencyByOpen); + //平仓所需资金 + BigDecimal closeNeed = new BigDecimal(getDate2.getLastPrice()).multiply(foreignCurrencyByClose); + + resp.setNeededPriceByOpen(openNeed); + resp.setNeededPriceByClose(closeNeed); + resp.setCreateTime(start); + resp.setUpdateTime(end); + resp.setExchangeRate1(new BigDecimal(getDate1.getLastPrice())); + resp.setExchangeRate2(new BigDecimal(getDate2.getLastPrice())); + return resp; + } + + //对冲后损益情况 + @Override + public CdAfterHedgingResp afterHedging(int userId) { + try { + CdAfterHedgingResp resp = new CdAfterHedgingResp(); + + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("user_id", userId); + //查询当前开仓信息 + CdFuturesConfigureEntity config = baseMapper.selectOne(queryWrapper); + + //开仓时间 + String startTime = DateUtils.dateForByStr(config.getCreateTime()); + //样本区间截止时间:开仓时间后三个月 + String endTime = DateUtils.getAfterMonth(startTime, 3); + + + QhscGangjiaosuoEntity qhEntity = qhscGangjiaosuoDao.selectById(config.getPid()); + + + //根据开仓时间-开仓时间后三个月查询 + QueryWrapper qhWarapper = new QueryWrapper<>(); + qhWarapper.eq("e_name", qhEntity.getEName()); + qhWarapper.ge("date", startTime).le("date", endTime); + + List rmb = qhscGangjiaosuoDao.selectList(qhWarapper); + String code = ""; + String name = ""; + + for (QhscGangjiaosuoEntity e : rmb) { + + String s = String.valueOf(e.getLastPrice()); + code += s + ","; + name = e.getName(); + } + code = code.substring(0, code.length() - 1); + + + //每手交易单位(合约面值) + Integer tradingUnit = setCompany(name); + + //confidenceLevel 置信水平 + String confidenceLevel = "0.95"; + //validity 持有效 + String validity = "90"; + //实验次数 + String experimentsNumber = "1000"; + + + //Z是传入进来的下单数*期货合约面值 + Integer Z = config.getEntrustNumber() * tradingUnit; + + + String afterBefore = ""; + String afterURL = ""; + String beforeURL = ""; + do { + //I为次数,t是天数,Z是传入进来的下单数*期货合约面值,arf为置信水平, + afterBefore = PyUtil.getCodeByPhoto(PyUtil.path, PyUtil.beforeAndAfter, experimentsNumber, validity, String.valueOf(Z), confidenceLevel, code); + + if (afterBefore.length() > 0) { + + String[] split = afterBefore.split(","); + String p1 = split[0]; + String p2 = split[1]; + afterURL = p2; + beforeURL = p1; + } + + } while (afterBefore.isEmpty()); + + + resp.setBeforeHedgingUrl(beforeURL); + resp.setAfterHedgingUrl(afterURL); + resp.setName(name.substring(0, 6)); + resp.setStartTime(startTime); + resp.setEndTime(endTime); + resp.setEffective(validity); + resp.setConfidenceLevel(confidenceLevel); + resp.setExperimentNum(experimentsNumber); + + + return resp; + } catch (IOException e) { + e.printStackTrace(); + log.error(e.getMessage()); + } + + + return null; + } + + @Override + public boolean delOpenPosition(int id, int userId) { + //删除开仓信息 + int flag = baseMapper.deleteById(id); + if (flag == 1) { + //先查看流水表要返回的金额 + + QueryWrapper wrapper = new QueryWrapper<>(); + wrapper.eq("user_id", userId); + wrapper.eq("transfer_type", "2"); + + CdBankStatementsEntity bankStatementsEntity = bankStatementsDao.selectOne(wrapper); + + + //删除账户流水表记录 + Map map = new HashMap<>(); + //删除类型为流水的 : (1:转账 2:流水) + map.put("transfer_type", "2"); + map.put("user_id", userId); + + bankStatementsDao.deleteByMap(map); + + + //头寸表记录 + //开仓平仓调仓记录表 + //修改用户账户今天为原来的 + + } + + return false; + } + + + //设置交易单位(合约面值) + public static Integer setCompany(String getName) { + try { + //每手交易单位 + Integer tradingUnit = 0; + //设置每手交易单位 + switch (getName.substring(0, 8)) { + case algorithm.AUDAndRMB: + tradingUnit = algorithm.AUDExchangeRMB; + break; + case algorithm.EURAndRMB: + tradingUnit = algorithm.EURExchangeRMB; + break; + case algorithm.JPYAndRMB: + tradingUnit = algorithm.JPYExchangeRMB; + break; + case algorithm.RMBAndUSD: + tradingUnit = algorithm.RMBExchangeUSD; + break; + case algorithm.USDAndRMB: + tradingUnit = algorithm.USDExchangeRMB; + break; + + } + return tradingUnit; + } catch (Exception e) { + e.printStackTrace(); + return -1; + } + } + + +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/CdPointChildrenServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/CdPointChildrenServiceImpl.java new file mode 100644 index 0000000..6c57194 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/CdPointChildrenServiceImpl.java @@ -0,0 +1,24 @@ +package com.msdw.tms.service.impl; + +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.dao.CdPointChildrenDao; +import com.msdw.tms.entity.CdPointChildren; +import com.msdw.tms.service.CdPointChildrenService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +/** + * @描述:判分点值子级表 服务类 + * @作者: Rong + * @日期: 2021-01-12 + */ +@Service +public class CdPointChildrenServiceImpl extends ServiceImpl implements CdPointChildrenService { + + @Autowired + private CdPointChildrenDao mapper; + +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/CdPointServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/CdPointServiceImpl.java new file mode 100644 index 0000000..e1d8479 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/CdPointServiceImpl.java @@ -0,0 +1,24 @@ +package com.msdw.tms.service.impl; + +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.dao.CdPointDao; +import com.msdw.tms.entity.CdPoint; +import com.msdw.tms.service.CdPointService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +/** + * @描述: 服务类 + * @作者: Rong + * @日期: 2021-01-11 + */ +@Service +public class CdPointServiceImpl extends ServiceImpl implements CdPointService { + + @Autowired + private CdPointDao dao; + +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/CdTransferWarehouseRecordServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/CdTransferWarehouseRecordServiceImpl.java new file mode 100644 index 0000000..9526148 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/CdTransferWarehouseRecordServiceImpl.java @@ -0,0 +1,366 @@ +package com.msdw.tms.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.trading_rules.algorithm; +import com.msdw.tms.dao.*; +import com.msdw.tms.entity.*; +import com.msdw.tms.service.CdTransferWarehouseRecordService; +import org.springframework.data.redis.core.RedisTemplate; +import org.springframework.stereotype.Service; + +import javax.annotation.Resource; +import java.math.BigDecimal; + +/** + * @描述:调仓记录表 服务类 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Service("cdTransferWarehouseRecordService") +public class CdTransferWarehouseRecordServiceImpl extends ServiceImpl implements CdTransferWarehouseRecordService { + + @Resource + private CdUserOptionAccountDao accountDao; + @Resource + private CdFuturesConfigureDao futuresConfigureDao; + @Resource + public QhscGangjiaosuoDao qhscGangjiaosuoDao; + + @Resource + public CdBankStatementsDao bankStatementsDao; + + @Resource + public CdTransferWarehouseRecordDao transferWarehouseRecordDao; + @Resource + public RedisTemplate template; + + /** + * 开仓买入 平仓卖出 + *

+ * 多头下单后,可以选择调仓 + * 调仓成功的前提是期货期权账号可用资金>=本次调仓所需要的资金,调仓下单成功,否则,提示可用资金不足,调仓失败。(完成) + *

+ *

+ *

+ * 2.如果减仓(减仓的数量不超过持仓中的多头数量),扣除保证金,按照最终的委托数量核算市值,浮动盈亏(同上公式)。 + * 市值=外汇期货合约价格*(原持仓量- 本次调仓量)*合约面值,浮动盈亏=(当天结算价—昨日结算价)*(原持仓量-本次调仓量)*合约面值。 + *

+ *

+ * 空头下单后,可以选择调仓,如果加仓,按照多头下单,扣除保证金,按照最终的委托数 + * 量核算市值,浮动盈亏。如果减仓(减仓的数量不超过持仓中的空头数量),扣除保证金, + * 按照最终的委托数量核算市值,浮动盈亏(同上公式)。 + */ + + //调仓操作 + @Override + public R add(CdFuturesConfigureEntity entity) { + + try { + String getDate = template.opsForValue().get("date:userId_" + entity.getUserId()); + + /**可交易数量(用户选择的买卖方向为开仓时候:查python表的持仓总量/用户选择的买卖方向为平仓时候:查自己的持仓总量)*/ + CdFuturesConfigureEntity old = futuresConfigureDao.selectById(entity.getId()); + QhscGangjiaosuoEntity qh = qhscGangjiaosuoDao.selectById(old.getPid()); + //查python的持仓总量 + BigDecimal getPyNumber = new BigDecimal(qh.getHava()); + //自己的持仓总量 + BigDecimal getMeNumber = old.getTradableNumber(); + + + //根据合约代码查询当天的汇率信息 + QueryWrapper queryWrapper2 = new QueryWrapper(); + queryWrapper2.eq("e_name", qh.getEName()); + //TODO 调仓时间为选中的时间 + queryWrapper2.eq("date", getDate); + QhscGangjiaosuoEntity getNow = qhscGangjiaosuoDao.selectOne(queryWrapper2); + if (getNow == null) { + return R.error("暂时没有" + getDate + "的外汇信息"); + } + //当前调仓输入的委托价 + double entrustPrice = entity.getEntrustPrice().doubleValue(); + + //查询合约当天外汇汇率价格 + double getNowPrice = getNow.getLastPrice().doubleValue(); + + + //根据合约名称设置每手交易单位(合约面值) + Integer tradingUnit = CdFuturesConfigureServiceImpl.setCompany(qh.getName()); + + //获取当前期权账户可用资金 + QueryWrapper getPrice = new QueryWrapper(); + getPrice.eq("user_id", entity.getUserId()); + //nowBalance:获取到当前期权账户可用资金 + BigDecimal nowBalance = accountDao.selectOne(getPrice).getBalance(); + + + //判断用户选择的是平仓还是开仓 (0开仓 1平仓) + switch (entity.getStatus()) { + case 0: + //0开仓表买入:用户选择的买卖方向为开仓时候:获取当前合约的持仓总量 + + + //市值:外汇期货合约价格*(原持仓量 + 本次调仓量)* 合约面值 + //BigDecimal getMarketValueByAddition = algorithm.getMarketValueByAddition(now.getLastPrice(), getMeNumber, BigDecimal.valueOf(entity.getEntrustNumber()), CdFuturesConfigureServiceImpl.setCompany(qh.getName())); + + //扣除保证金(保证金 = 外汇期货合约价格 * 委托数量 * 合约面值 * 保证金比例) + BigDecimal bzj = algorithm.freezingMargin(getNow.getLastPrice(), entity.getTradableNumber(), new BigDecimal(tradingUnit), qh.getName().substring(0, 8)); + + + //TODO 总共要扣除的金额 = 需要扣除的保证金 + 所需资金 + + //所需资金 = 委托价格 * 委托数量(手数) * 每手交易单位 + BigDecimal funding_needed = algorithm.getFundsRequired(entity.getEntrustPrice(), BigDecimal.valueOf(entity.getEntrustNumber()), tradingUnit); + + + /** + * 期货的开仓成交规则: + * 多方情况下,委托价格大于等于最新价,以最新价成交; + * 空方情况下,委托价格小于等于最新价,以最新价成交。 + */ + // tODO 暂时注释 + //交易方向(0空头,1多头) + switch (old.getTradingDirection()) { + case 0: + //空方情况下:委托价格小于等于最新价,以最新价成交 + // entrustPrice > newestPrice + if (entrustPrice > getNowPrice) { + return R.error("开仓失败,委托价格大于最新价"); + } + break; + case 1: + //多方情况下:委托价格大于等于最新价,以最新价成交 + // entrustPrice < newestPrice + if (entrustPrice < getNowPrice) { + return R.error("开仓失败,委托价格小于最新价"); + } + break; + + } + //对当前头寸表 需要修改的操作: + // 交易方向?、 + CdFuturesConfigureEntity modifyByCondition = new CdFuturesConfigureEntity(); + + //最新价格:开仓时候 最新价格以当天汇率价格 + modifyByCondition.setNewestPrice(getNow.getLastPrice()); + + //委托价格 当前输入的价格 + modifyByCondition.setEntrustPrice(entity.getEntrustPrice()); + + //委托数量 + modifyByCondition.setEntrustNumber(entity.getEntrustNumber()); + + //预冻结保证金 = 外汇期货合约价格 * 合约面值 * 手数 * 保证金比例 + BigDecimal getPreFreezingMargin = algorithm.getPreFreezingMargin(qh.getLastPrice(), new BigDecimal(tradingUnit), qh.getName().substring(0, 8)); + modifyByCondition.setPreFreezingMargin(getPreFreezingMargin); + + //预冻结手续费 = 委托数量(手)* 费用标准 + BigDecimal getPreFreezingServiceCharge = algorithm.getPreFreezingServiceCharge(new BigDecimal(entity.getEntrustNumber())); + modifyByCondition.setPreFreezingFrozen(getPreFreezingServiceCharge); + + + //frozen_price:冻结资金 = 预计冻结保证金 + 预冻结手续费 + BigDecimal frozen_price = getPreFreezingMargin.add(getPreFreezingServiceCharge); + modifyByCondition.setFrozenPrice(frozen_price); + + //设置所需资金 + modifyByCondition.setFundingNeeded(funding_needed); + + + //计算合计操作的总持仓量 = 自己的持仓总量 + 本次要加仓的数量 + BigDecimal getTotal = getMeNumber.add(BigDecimal.valueOf(entity.getEntrustNumber())); + //可交易数量 = 之前的委托数量 + 本次委托数量 + modifyByCondition.setTradableNumber(getTotal); + + //委托数量 + modifyByCondition.setEntrustNumber(getTotal.intValue()); + // 浮动盈亏=(当天结算价—昨日结算价)* 委托数量 * 合约面值(1份合约=一手,一手=?吨) + BigDecimal floatPl = algorithm.getFloatingPL(getNow.getLastPrice(), getNow.getDayUp(), new BigDecimal(entity.getEntrustNumber()), CdFuturesConfigureServiceImpl.setCompany(qh.getName())); + //设置浮动盈亏 + modifyByCondition.setFloatingPl(floatPl); + //设置为开仓 + modifyByCondition.setStatus(0); + + //总共要扣除的金额 = 需要扣除的保证金 + 所需资金 + BigDecimal TotalAmount = funding_needed.add(bzj); + //判断当前可用余额是否足以扣除本次交易金额 + if (nowBalance.doubleValue() < TotalAmount.doubleValue()) { + return R.error("可用资金不足,开仓失败"); + } + + //开仓时间 + modifyByCondition.setCreateTime(getDate); + modifyByCondition.setId(entity.getId()); + int modify = futuresConfigureDao.updateById(modifyByCondition); + + + //modifyPrice:要做修改的金额(要做修改的金额 = 当前账户可用资金 - 要扣除的金额) + BigDecimal modifyPrice = nowBalance.subtract(TotalAmount); + + + //对用户期权账户修改金额操作 + UpdateWrapper updateWrapper2 = new UpdateWrapper(); + updateWrapper2.set("balance", modifyPrice); + updateWrapper2.eq("user_id", entity.getUserId()); + int updateAssets = accountDao.update(null, updateWrapper2); + + + //添加期权账户流水记录 + CdBankStatementsEntity bankStatements2 = new CdBankStatementsEntity(); + bankStatements2.setType(2);//类型(1银行账户 2期货账户) + bankStatements2.setCoinType("人民币"); + bankStatements2.setBeforeAmount(nowBalance);//改变前金额 + bankStatements2.setAmountMoney(TotalAmount);//这次操作的金额 + bankStatements2.setAfterAmount(modifyPrice);//改变后的金额 + bankStatements2.setTransferType(2);//流水类型(1:转账 2:流水) + bankStatements2.setCreateTime(getDate); + bankStatements2.setUserId(entity.getUserId()); + bankStatements2.setRemark("开仓"); + int addBankStatements2 = bankStatementsDao.insert(bankStatements2); + + + //往开仓表添加开仓记录 + //往流水记录表添加开仓流水记录 + CdTransferWarehouseRecordEntity record = new CdTransferWarehouseRecordEntity(); + //开仓表id + record.setCid(entity.getId()); + record.setUserId(entity.getUserId()); + record.setDirection(0);//买卖方向(0开仓 1平仓) + record.setType(0);//委托类型(0市价 1限价) + record.setNumber(entity.getTradableNumber());//委托手数 + record.setBalance(TotalAmount);//市值>>>金额 + record.setCreateTime(getDate);//TODO 调仓时开仓时间为选中的时间 + int addRe = transferWarehouseRecordDao.insert(record); + + if (modify <= 0 && updateAssets <= 0 && addBankStatements2 <= 0 && addRe <= 0) { + return R.error("调仓失败"); + } + + break; + + case 1: + /* 1平仓表示卖出 */ + + Integer num = entity.getEntrustNumber(); + + //当前可交易数量 - 我要平仓的数据 + Integer getTradableNumber = old.getTradableNumber().intValue(); + if (getTradableNumber - num < 1) { + return R.error("委托数量不能是持仓总量,至少留一手~"); + } + + + /** + * 期货的平仓成交规则: + * configure.getTradingDirection() : 交易方向(0空头,1多头) + * 空方情况下,可交易数量为空方合约持仓总量,委托价格大于等于最新价,以最新价成交。 + * 多方情况下,可交易数量为多方合约持仓总量,委托价格小于等于最新价,以最新价成交; + */ + // TODO 暂时注释 + switch (old.getTradingDirection()) { + case 0: + //空方情况下:委托价格大于等于最新价,以最新价成交。 + if (entrustPrice < getNowPrice) { + return R.error("平仓失败,委托价格小于最新价"); + } + break; + case 1: + //多方情况下:委托价格小于等于最新价,以最新价成交; + if (entrustPrice > getNowPrice) { + return R.error("平仓失败,委托价格大于最新价"); + } + break; + + } + + + //外汇期货盈亏=(平仓时的外汇汇率价格(等同于当天合约的最新价)-下单时外汇汇率价格(开仓时候的委托价)) * 委托数量 * 合约面值。 + BigDecimal pl = algorithm.get_wh_pl(BigDecimal.valueOf(getNowPrice), old.getEntrustPrice(), new BigDecimal(old.getEntrustNumber()), new BigDecimal(tradingUnit)); + + //getMarketValue:市值 = 外汇期货合约价格 * 委托数量 * 合约面值 + BigDecimal getMarketValue = algorithm.getMarketValue(BigDecimal.valueOf(getNowPrice), entity.getEntrustNumber(), tradingUnit); + + + //根据id查询开仓后的信息 + //CdFuturesConfigureEntity configure = futuresConfigureDao.selectById(entity.getId()); + + + //对当前头寸表 需要修改的操作: + // 交易方向?、 + CdFuturesConfigureEntity modifyByCondition1 = new CdFuturesConfigureEntity(); + + //设置平仓时间 + modifyByCondition1.setUpdateTime(old.getTradeTime()); + //设置状态(0.开仓 1平仓) + modifyByCondition1.setStatus(1); + //设置平仓价格:市场当前价 + modifyByCondition1.setCloseRate(getMarketValue); + //设置盈亏 + modifyByCondition1.setFloatingPl(pl); + + modifyByCondition1.setId(entity.getId()); + //修改设置可交易量 + modifyByCondition1.setTradableNumber(BigDecimal.valueOf(old.getEntrustNumber()).subtract(BigDecimal.valueOf(entity.getEntrustNumber()))); + + //平仓持仓量 等于持仓量减去本次下单手数 + modifyByCondition1.setEntrustNumber(old.getEntrustNumber() - entity.getEntrustNumber()); + + int modify1 = futuresConfigureDao.updateById(modifyByCondition1); + + + //改变后的金额 + BigDecimal newPrice = nowBalance.add(getMarketValue); + + + //做修改账户金额操作 + UpdateWrapper modifyBypc = new UpdateWrapper(); + //平仓后加金额 = 当前可用余额 + 市值 + modifyBypc.set("balance", newPrice); + modifyBypc.eq("user_id", entity.getUserId()); + int updateAsset = accountDao.update(null, modifyBypc); + + + //往流水记录表添加平仓流水记录 + CdTransferWarehouseRecordEntity record1 = new CdTransferWarehouseRecordEntity(); + //开仓表id + record1.setCid(entity.getId()); + record1.setUserId(entity.getUserId()); + record1.setDirection(1);//买卖方向(0开仓 1平仓) + record1.setType(0);//委托类型(0市价 1限价) + record1.setNumber(entity.getTradableNumber());//委托手数 + record1.setBalance(getMarketValue);//市值>>>金额 + record1.setCreateTime(getDate); + int addTransferWarehouseRecor = transferWarehouseRecordDao.insert(record1); + + + //添加期权账户流水记录 + CdBankStatementsEntity bankStatements = new CdBankStatementsEntity(); + bankStatements.setType(2);//类型(1银行账户 2期货账户) + bankStatements.setCoinType("人民币"); + bankStatements.setBeforeAmount(nowBalance);//改变前金额 + bankStatements.setAmountMoney(getMarketValue);//这次操作的金额 + bankStatements.setAfterAmount(newPrice);//改变后的金额 + bankStatements.setTransferType(2);//流水类型(1:转账 2:流水) + bankStatements.setCreateTime(getDate); + bankStatements.setRemark("平仓"); + bankStatements.setUserId(entity.getUserId()); + int addBankStatements = bankStatementsDao.insert(bankStatements); + + + if (modify1 <= 0 && updateAsset <= 0 && addBankStatements <= 0 && addTransferWarehouseRecor <= 0) { + return R.error("平仓失败"); + } + break; + } + } catch (Exception e) { + e.printStackTrace(); + return R.error(); + } + return R.ok(); + } +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/CdUserAssetsServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/CdUserAssetsServiceImpl.java new file mode 100644 index 0000000..2d57ef9 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/CdUserAssetsServiceImpl.java @@ -0,0 +1,66 @@ +package com.msdw.tms.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.dao.CdUserAssetsDao; +import com.msdw.tms.entity.CdUserAssetsEntity; +import com.msdw.tms.service.CdUserAssetsService; +import org.springframework.stereotype.Service; + +import javax.annotation.Resource; + +/** + * @描述:用户银行账户 服务类 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Service("CdUserAssetsService") +public class CdUserAssetsServiceImpl extends ServiceImpl implements CdUserAssetsService { + + @Resource + private CdUserAssetsDao dao; + + + //插入银行信息账户信息以及期权账户信息 + @Override + public int addInfo(CdUserAssetsEntity userAssetsEntity) { + try { + int code = baseMapper.insert(userAssetsEntity); + return code; + } catch (Exception e) { + e.printStackTrace(); + return -1; + } + } + + @Override + public CdUserAssetsEntity userAssetsByUerId(long userId) { + try { + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", userId); + CdUserAssetsEntity optionAccount = optionAccount = baseMapper.selectOne(queryWrapper); + return optionAccount; + } catch (Exception e) { + e.printStackTrace(); + return null; + } + } + + @Override + public int modify(long userId) { + try { + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", userId); + CdUserAssetsEntity userAssets = new CdUserAssetsEntity(); + userAssets.setUserId(userId); + int update = baseMapper.update(userAssets, queryWrapper); + return update; + } catch (Exception e) { + e.printStackTrace(); + return -1; + } + } +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/CdUserOptionAccountServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/CdUserOptionAccountServiceImpl.java new file mode 100644 index 0000000..3e7bcb8 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/CdUserOptionAccountServiceImpl.java @@ -0,0 +1,73 @@ +package com.msdw.tms.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.dao.CdUserOptionAccountDao; +import com.msdw.tms.entity.CdUserOptionAccountEntity; +import com.msdw.tms.service.CdUserOptionAccountService; +import org.springframework.stereotype.Service; + +import javax.annotation.Resource; +import java.math.BigDecimal; + +/** + * @描述:期货期权账户 服务类 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Service("cdUserOptionAccountService") +public class CdUserOptionAccountServiceImpl extends ServiceImpl implements CdUserOptionAccountService { + + @Resource + private CdUserOptionAccountDao dao; + + @Override + public int addInfoByOptionAccount(CdUserOptionAccountEntity userOptionAccount) { + + + try { + int code = baseMapper.insert(userOptionAccount); + return code; + } catch (Exception e) { + e.printStackTrace(); + return -1; + } + } + + //查看期权账户信息 + @Override + public CdUserOptionAccountEntity optionAccountByUerId(long userId) { + try { + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", userId); + CdUserOptionAccountEntity optionAccount = baseMapper.selectOne(queryWrapper); + //总资金 = 资产市值 + 占用保证金 + 冻结资金 + 可用资金 + BigDecimal totalAssets = optionAccount.getDepositBalance().add(optionAccount.getDepositBalance().add(optionAccount.getBalance().add(optionAccount.getForzenBalance()))); + optionAccount.setTotalAssets(totalAssets); + return optionAccount; + } catch (Exception e) { + e.printStackTrace(); + return null; + } + + + } + + @Override + public int modify(long userId) { + try { + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("user_id", userId); + CdUserOptionAccountEntity userAssets = new CdUserOptionAccountEntity(); + userAssets.setUserId(userId); + int update = baseMapper.update(userAssets, queryWrapper); + return update; + } catch (Exception e) { + e.printStackTrace(); + return -1; + } + } +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/QhscGangjiaosuoServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/QhscGangjiaosuoServiceImpl.java new file mode 100644 index 0000000..0f1dee6 --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/QhscGangjiaosuoServiceImpl.java @@ -0,0 +1,43 @@ +package com.msdw.tms.service.impl; + +import com.baomidou.dynamic.datasource.annotation.DS; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.common.utils.PageUtils; +import com.msdw.tms.dao.QhscGangjiaosuoDao; +import com.msdw.tms.entity.QhscGangjiaosuoEntity; +import com.msdw.tms.service.QhscGangjiaosuoService; +import org.springframework.stereotype.Service; + +import java.util.List; + +/** + * @描述: 服务类 + * @作者: Rong + * @日期: 2020-10-10 + */ +@Service("qhscGangjiaosuoService") +@DS("pydb") +public class QhscGangjiaosuoServiceImpl extends ServiceImpl implements QhscGangjiaosuoService { + + + @Override + public PageUtils queryPageByQh(Integer page, Integer size, String name, String date) { + //分页对象 + Page page1 = new Page(page, size); + IPage list = this.baseMapper.queryPageByQh(page1, name, date); + PageUtils pageUtils = new PageUtils(list); + return pageUtils; + } + + @Override + public List getName() { + return this.baseMapper.getName(); + } + + +} + + + diff --git a/src/main/java/com/msdw/tms/service/impl/WhscRenminbipinzhongServiceImpl.java b/src/main/java/com/msdw/tms/service/impl/WhscRenminbipinzhongServiceImpl.java new file mode 100644 index 0000000..024183c --- /dev/null +++ b/src/main/java/com/msdw/tms/service/impl/WhscRenminbipinzhongServiceImpl.java @@ -0,0 +1,348 @@ +package com.msdw.tms.service.impl; + +import cn.afterturn.easypoi.excel.ExcelExportUtil; +import cn.afterturn.easypoi.excel.entity.ExportParams; +import com.baomidou.dynamic.datasource.annotation.DS; +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.msdw.tms.common.utils.PageUtils; +import com.msdw.tms.common.utils.R; +import com.msdw.tms.common.utils.py.PyUtil; +import com.msdw.tms.dao.QhscGangjiaosuoDao; +import com.msdw.tms.dao.WhscRenminbipinzhongDao; +import com.msdw.tms.entity.QhscGangjiaosuoEntity; +import com.msdw.tms.entity.WhscRenminbipinzhongEntity; +import com.msdw.tms.entity.vo.CdCreatePhotoVo; +import com.msdw.tms.service.WhscRenminbipinzhongService; +import org.apache.commons.lang.StringUtils; +import org.apache.poi.ss.usermodel.Workbook; +import org.springframework.data.redis.core.RedisTemplate; +import org.springframework.stereotype.Service; + +import javax.annotation.Resource; +import javax.servlet.http.HttpServletResponse; +import java.io.IOException; +import java.net.URLEncoder; +import java.util.Iterator; +import java.util.List; + +/** + * @描述:外汇市场人民币品种 服务类 + * @作者: Rong + * @日期: 2020-09-28 + */ +//@Slf4j +@Service("whscRenminbipinzhongService") +@DS("pydb") +public class WhscRenminbipinzhongServiceImpl extends ServiceImpl implements WhscRenminbipinzhongService { + + @Resource + public RedisTemplate template; + + @Resource + public QhscGangjiaosuoDao qhscGangjiaosuoDao; + + //导出Excel + @Override + public R exportProjectRecord(HttpServletResponse response, String name, String startTime, String endTime) throws Exception { + + + try { + String strSql = " "; + int i = 0;//定义是否第一次循环 + QueryWrapper queryWrapper = new QueryWrapper<>(); + //查询币种名称 + List getRmb = qhscGangjiaosuoDao.getName(); + Iterator it = getRmb.iterator(); + + + if (!StringUtils.isEmpty(name)) { + queryWrapper.like("name", name + "人民币中间价"); + } + + if (!StringUtils.isEmpty(startTime) && !StringUtils.isEmpty(endTime)) { + queryWrapper.ge("date", startTime).le("date", endTime); + } + + //如果币种 时间都为空导出全部 + if (StringUtils.isEmpty(name) && StringUtils.isEmpty(startTime) && StringUtils.isEmpty(endTime)) { + //int i = 0;//定义是否第一次循环 + while (it.hasNext()) { + String x = it.next().getCurrency(); + if (x.equals("人民币")) { + it.remove(); + continue; + } + + if (i == 0) { + strSql += " WHERE ( name like concat('" + x + "%人民币中间价')"; + i++; + } + + //queryWrapper.like("name", "中间价"); + strSql += " or name like concat('" + x + "%人民币中间价')"; + //queryWrapper.last(" and name like concat('%"+x+"中间价')"); + } + //strSql += ")"; + queryWrapper.last(strSql); + //参数一是当前页,参数二是每页个数 + } + + //如果不选择币种时候默认查询全部的 + if (StringUtils.isEmpty(name)) { + + //int i = 0;//定义是否第一次循环 + while (it.hasNext()) { + String x = it.next().getCurrency(); + if (x.equals("人民币")) { + it.remove(); + continue; + } + + if (i == 0) { + strSql += " and ( name like concat('" + x + "%人民币中间价')"; + i++; + } + + //queryWrapper.like("name", "中间价"); + strSql += " or name like concat('" + x + "%人民币中间价')"; + //queryWrapper.last(" and name like concat('%"+x+"中间价')"); + } + strSql += ")"; + queryWrapper.last(strSql); + //参数一是当前页,参数二是每页个数 + } + + + //list + List list = this.list(queryWrapper); + if (list.size() == 0) { + return R.ok("暂无此数据,换个时间试试吧"); + } + + + // 告诉浏览器用什么软件可以打开此文件 + response.setHeader("content-Type", "application/vnd.ms-excel"); + // 下载文件的默认名称 + response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("汇率历史行情数据", "UTF-8") + ".xls"); + //编码 + response.setCharacterEncoding("UTF-8"); + Workbook workbook = ExcelExportUtil.exportExcel(new ExportParams(), WhscRenminbipinzhongEntity.class, list); + workbook.write(response.getOutputStream()); + return null; + } catch (IOException e) { + e.printStackTrace(); + return R.error("出现异常:" + e.getMessage()); + + } + + + } + + + /** + * 步骤二:当日外汇行情 + * + * @param pageNum + * @param pageSize + * @param name + * @param date + * @return + */ + @Override + public PageUtils queryPageByRecode(Integer pageNum, Integer pageSize, String name, String date, Integer userId) { + + if (date != null) { + template.opsForValue().set("date:userId_" + userId, date); + } + + //分页对象 + QueryWrapper queryWrapper = new QueryWrapper(); + //查询币种名称 + List getRmb = qhscGangjiaosuoDao.getName(); + Iterator it = getRmb.iterator(); + //判断条件值是否为空,如果不为空,拼接条件 + /*if (!StringUtils.isEmpty(date)) { + queryWrapper.eq("date", date); + } + if (!StringUtils.isEmpty(name)) { + queryWrapper.like("name", name + "人民币中间价"); + }*/ + + + if (!StringUtils.isEmpty(name) && !StringUtils.isEmpty(date)) { + queryWrapper.eq("date", date); + queryWrapper.like("name", name + "人民币中间价"); + } + String strSql = " "; + int i = 0;//定义是否第一次循环 + + //日期不为空 名称为空时 + if (!StringUtils.isEmpty(date) && StringUtils.isEmpty(name)) { + strSql = "where 1 = 1"; + + while (it.hasNext()) { + String x = it.next().getCurrency(); + if (x.equals("人民币")) { + it.remove(); + continue; + } + + if (i == 0) { + strSql += " and name like concat('%" + x + "人民币中间价%') and DATE_FORMAT(date, '%Y-%m-%d') = '" + date + "'"; + + + i++; + } else if (i == 1) { + strSql += " or name like concat('%" + x + "人民币中间价%') and DATE_FORMAT(date, '%Y-%m-%d') = '" + date + "'"; + } + + + } + //strSql += ")"; + queryWrapper.last(strSql); + //参数一是当前页,参数二是每页个数 + } + + //名称为空时 + if (StringUtils.isEmpty(name) && StringUtils.isEmpty(date)) { + strSql = "where 1=1"; + //查询币种名称 + //String strSql = " "; + //int i = 0;//定义是否第一次循环 + while (it.hasNext()) { + String x = it.next().getCurrency(); + if (x.equals("人民币")) { + it.remove(); + continue; + } + + if (i == 0) { + strSql += " and name like concat('" + x + "%人民币中间价')"; + i++; + } + + //queryWrapper.like("name", "中间价"); + strSql += " or name like concat('" + x + "%人民币中间价')"; + //queryWrapper.last(" and name like concat('%"+x+"中间价')"); + } + //strSql += ")"; + queryWrapper.last(strSql); + //参数一是当前页,参数二是每页个数 + } + + Page page = new Page(pageNum, pageSize); + page = this.page(page, queryWrapper); + return new PageUtils(page); + } + + /** + * 根据选择的模拟法传值获取相应数据 + * + * @param name + * @param startTime 样本区间开始时间 + * @param endTime 样本区间结束时间 + * @param confidenceLevel 置信水平 + * @param validity 持有效 + * @param type 类型(0.正态VaR 1.历史模拟法 2.蒙特卡洛模拟法) + * @param experimentsNumber 实验次数 + * @return + */ + @Override + public R getInfobyId(String name, String startTime, String endTime, String confidenceLevel, String validity, Integer type, String experimentsNumber) { + + try { + + if (startTime.equals(endTime)) { + return R.error("样本区间时间不能为同一天!"); + } + + QueryWrapper wrapper = new QueryWrapper(); + wrapper.like("name", name); + wrapper.last("limit 0,1"); + + //获取代码 + String e_name = baseMapper.selectOne(wrapper).geteName(); + + + //根据代码样本日期查询 + QueryWrapper queryWrapper = new QueryWrapper(); + queryWrapper.eq("e_name", e_name); + if (startTime != null && endTime != null) { + queryWrapper.ge("date", startTime).le("date", endTime); + } + List rmb = baseMapper.selectList(queryWrapper); + + if (rmb.size() == 0) { + return R.error("暂时没有该区间的数据!"); + } else if (rmb.size() < 2) { + return R.error("暂时没有该区间的数据!"); + } else { + + String code = ""; + for (WhscRenminbipinzhongEntity e : rmb) { + + String s = String.valueOf(e.getLastPrice()); + code += s + ","; + } + code = code.substring(0, code.length() - 1); + + CdCreatePhotoVo vo = new CdCreatePhotoVo(); + + /** + * + * getVar 置信水平(小数) 天数 列表 + * getVarByHistory 置信水平(小数) 天数 列表 + * photo 次数 天数 列表 + */ + //(0.正态VaR 1.历史模拟法 2.蒙特卡洛模拟法) + switch (type) { + case 0: + String getVar = PyUtil.getVarCode(PyUtil.path, PyUtil.pyPathByVar, confidenceLevel, validity, code); + vo.setVar(getVar); + return R.ok().put("data", vo); + case 1: + String getVarByHistory = PyUtil.getCodeByHis(PyUtil.path, PyUtil.historyPath, confidenceLevel, validity, code); + vo.setVar(getVarByHistory); + return R.ok().put("data", vo); + + case 2: + + String buffer = PyUtil.getCodeByPhoto(PyUtil.path, PyUtil.pyPathByPhoto, experimentsNumber, validity, code); + if (buffer.length() > 0) { + + String[] split = buffer.split(","); + String p1 = split[0]; + String p2 = split[1]; + vo.setPhoto_url1(p1); + vo.setPhoto_url2(p2); + return R.ok().put("data", vo); + } + default: + return R.error("参数异常"); + } + } + + + } catch (IOException e) { + e.printStackTrace(); + return R.error(e.getMessage()); + } + } + + /** + * 汇率类型 + * + * @return + */ + @Override + public List getExchangeRate() { + return baseMapper.getExchangeRate(); + } + + +} + + + diff --git a/src/main/resources/aliyun.properties b/src/main/resources/aliyun.properties new file mode 100644 index 0000000..acb5672 --- /dev/null +++ b/src/main/resources/aliyun.properties @@ -0,0 +1,6 @@ +oss.endpoint=oss-cn-shenzhen.aliyuncs.com +oss.accessKeyId=LTAIHIkGqaILObBm +oss.secretAccessKey=QDTxKMrfDPeJ3bsr3AqjYHwnlL6PdM +oss.bucketName=liuwanr +oss.sufferUrl=http://liuwanr.oss-cn-shenzhen.aliyuncs.com/ +user.userAvatars=userAvatars \ No newline at end of file diff --git a/src/main/resources/application.yml b/src/main/resources/application.yml new file mode 100644 index 0000000..f1c6df2 --- /dev/null +++ b/src/main/resources/application.yml @@ -0,0 +1,61 @@ +spring: + jackson: + default-property-inclusion: non_null + datasource: + dynamic: + primary: mydb #设置默认的数据源或者数据源组,默认值即为master + strict: false #设置严格模式,默认false不启动. 启动后在未匹配到指定数据源时候回抛出异常,不启动会使用默认数据源. + datasource: + mydb: + url: jdbc:mysql://localhost:3306/sichuan_subsystem?useUnicode=true&characterEncoding=UTF-8&useSSL=false&serverTimezone=Asia/Shanghai + username: root + password: e63aed97aaa1b41f + driver-class-name: com.mysql.cj.jdbc.Driver + pydb1: + url: jdbc:mysql://rm-wz95oz7m93cejtuko4o.mysql.rds.aliyuncs.com:3306/financial?autoReconnect=true&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=GMT%2b8 + username: super + password: super888 + driver-class-name: com.mysql.cj.jdbc.Driver + pydb: + url: jdbc:mysql://localhost:3306/py_test?autoReconnect=true&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=GMT%2b8 + username: root + password: e63aed97aaa1b41f + driver-class-name: com.mysql.cj.jdbc.Driver + + # redis: + # host: www.liuwanr.cn + # port: 6379 + # password: huoran + redis: + host: 127.0.0.1 + port: 6379 + password: dqredis123456 +mybatis-plus: + configuration: + log-impl: org.apache.ibatis.logging.stdout.StdOutImpl # 这个配置会将执行的sql打印出来,在开发或测试的时候可以用 + mapper-locations: classpath:/mapper/**/*.xml + global-config: + db-config: + id-type: auto + logic-delete-field: is_del + logic-delete-value: 0 + logic-not-delete-value: 1 +server: + port: 8080 + servlet: + context-path: /sichuan + +tms: + thread: + core-size: 20 + max-size: 200 + keep-alive-time: 10 + + +logging: + level: + org: + springframework: + boot: + autoconfigure: + logging: info \ No newline at end of file diff --git a/src/main/resources/excel-template/实验报告导出.xlsx b/src/main/resources/excel-template/实验报告导出.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..d979a8c26fb8ab87a279613f536e0bb34a728317 GIT binary patch literal 10295 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 literal 0 HcmV?d00001 diff --git a/src/main/resources/excel-template/实验报告导出模板.xlsx b/src/main/resources/excel-template/实验报告导出模板.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..03a2749edfaaab05b88f22783a4fd994bbe59580 GIT binary patch literal 10365 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

{H-h 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 JiTGLj{{X|);5+~T literal 0 HcmV?d00001 diff --git a/src/main/resources/excel-template/试题导入失败数据导出模板.xlsx b/src/main/resources/excel-template/试题导入失败数据导出模板.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..042244a71e2163d493e44eb7f863d06ac212c811 GIT binary patch literal 10358 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 GcmD^@Jmt#( literal 0 HcmV?d00001 diff --git a/src/main/resources/excel-template/试题导入模板.xlsx b/src/main/resources/excel-template/试题导入模板.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..7cc72303e7167eaaaa0cc247893d750072923287 GIT binary patch literal 11147 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

0d&T& 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 C8l7VR literal 0 HcmV?d00001 diff --git a/src/main/resources/logback-spring.xml b/src/main/resources/logback-spring.xml new file mode 100644 index 0000000..d09983d --- /dev/null +++ b/src/main/resources/logback-spring.xml @@ -0,0 +1,47 @@ + + + + + + + + + + + %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n + utf8 + + + + + + + + ${LOG_HOME}/tms.%d{yyyy-MM-dd}.log + + + %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n + + + + + + + 0 + + 512 + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/CdBankStatementsMapper.xml b/src/main/resources/mapper/tms/CdBankStatementsMapper.xml new file mode 100644 index 0000000..f37343b --- /dev/null +++ b/src/main/resources/mapper/tms/CdBankStatementsMapper.xml @@ -0,0 +1,23 @@ + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/CdCacheMapper.xml b/src/main/resources/mapper/tms/CdCacheMapper.xml new file mode 100644 index 0000000..a8bc715 --- /dev/null +++ b/src/main/resources/mapper/tms/CdCacheMapper.xml @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/CdExperimentalMapper.xml b/src/main/resources/mapper/tms/CdExperimentalMapper.xml new file mode 100644 index 0000000..d69908d --- /dev/null +++ b/src/main/resources/mapper/tms/CdExperimentalMapper.xml @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/CdFuturesConfigureMapper.xml b/src/main/resources/mapper/tms/CdFuturesConfigureMapper.xml new file mode 100644 index 0000000..f91f644 --- /dev/null +++ b/src/main/resources/mapper/tms/CdFuturesConfigureMapper.xml @@ -0,0 +1,28 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/CdPointChildrenMapper.xml b/src/main/resources/mapper/tms/CdPointChildrenMapper.xml new file mode 100644 index 0000000..931b929 --- /dev/null +++ b/src/main/resources/mapper/tms/CdPointChildrenMapper.xml @@ -0,0 +1,13 @@ + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/CdPointMapper.xml b/src/main/resources/mapper/tms/CdPointMapper.xml new file mode 100644 index 0000000..f5346b2 --- /dev/null +++ b/src/main/resources/mapper/tms/CdPointMapper.xml @@ -0,0 +1,12 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/CdTransferWarehouseRecordMapper.xml b/src/main/resources/mapper/tms/CdTransferWarehouseRecordMapper.xml new file mode 100644 index 0000000..2c5c3a7 --- /dev/null +++ b/src/main/resources/mapper/tms/CdTransferWarehouseRecordMapper.xml @@ -0,0 +1,18 @@ + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/CdUserAssetsMapper.xml b/src/main/resources/mapper/tms/CdUserAssetsMapper.xml new file mode 100644 index 0000000..32587c7 --- /dev/null +++ b/src/main/resources/mapper/tms/CdUserAssetsMapper.xml @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/CdUserOptionAccountMapper.xml b/src/main/resources/mapper/tms/CdUserOptionAccountMapper.xml new file mode 100644 index 0000000..6685d25 --- /dev/null +++ b/src/main/resources/mapper/tms/CdUserOptionAccountMapper.xml @@ -0,0 +1,16 @@ + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/QhscGangjiaosuoMapper.xml b/src/main/resources/mapper/tms/QhscGangjiaosuoMapper.xml new file mode 100644 index 0000000..ade6446 --- /dev/null +++ b/src/main/resources/mapper/tms/QhscGangjiaosuoMapper.xml @@ -0,0 +1,49 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/mapper/tms/WhscRenminbipinzhongMapper.xml b/src/main/resources/mapper/tms/WhscRenminbipinzhongMapper.xml new file mode 100644 index 0000000..2f3abf2 --- /dev/null +++ b/src/main/resources/mapper/tms/WhscRenminbipinzhongMapper.xml @@ -0,0 +1,24 @@ + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/main/resources/sql/data.sql b/src/main/resources/sql/data.sql new file mode 100644 index 0000000..410d162 --- /dev/null +++ b/src/main/resources/sql/data.sql @@ -0,0 +1,115 @@ +/* +Navicat MySQL Data Transfer + +Source Server : localhost +Source Server Version : 80020 +Source Host : localhost:3306 +Source Database : msdw_tms + +Target Server Type : MYSQL +Target Server Version : 80020 +File Encoding : 65001 + +Date: 2020-08-26 18:02:57 +*/ + +SET FOREIGN_KEY_CHECKS=0; + +-- ---------------------------- +-- Table structure for tms_evaluation_rules +-- ---------------------------- +DROP TABLE IF EXISTS `tms_evaluation_rules`; +CREATE TABLE `tms_evaluation_rules` ( + `id` int NOT NULL AUTO_INCREMENT COMMENT '主键', + `evaluation_type` int DEFAULT NULL COMMENT '测评类型:1:随机类型,0:自定义类型', + `duration` int DEFAULT NULL COMMENT '测评时长,单位:分钟', + `question_num` int DEFAULT NULL COMMENT '测评总题数', + `single_num` int DEFAULT NULL COMMENT '单选题数', + `is_single_enable` int DEFAULT '1' COMMENT '只在类型为自定义类型时才启用:1:启用,0:不启用,默认1启用', + `multiple_num` int DEFAULT NULL COMMENT '多选题数', + `is_multiple_enable` int DEFAULT '1' COMMENT '只在类型为自定义类型时才启用:1:启用,0:不启用,默认1启用', + `judgment_num` int DEFAULT NULL COMMENT '判断题数', + `is_judgment_enable` int DEFAULT '1' COMMENT '只在类型为自定义类型时才启用:1:启用,0:不启用,默认1启用', + PRIMARY KEY (`id`) +) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci COMMENT='测评规则信息记录,只记录一条信息'; + +-- ---------------------------- +-- Records of tms_evaluation_rules +-- ---------------------------- +INSERT INTO `tms_evaluation_rules` VALUES ('1', '1', '10', '0', '0', '1', '0', '1', '0', '1'); + +-- ---------------------------- +-- Table structure for tms_questions +-- ---------------------------- +DROP TABLE IF EXISTS `tms_questions`; +CREATE TABLE `tms_questions` ( + `id` int NOT NULL AUTO_INCREMENT COMMENT '自增主键', + `question_type` int DEFAULT NULL COMMENT '题型:1、单选题,2、多选题,3、判断题', + `question_stem` text COMMENT '题干信息', + `option_a` text COMMENT 'A选项内容', + `option_b` text COMMENT 'B选项内容', + `option_c` text COMMENT 'C选项内容', + `option_d` text COMMENT 'D选项内容', + `option_e` text COMMENT 'E选项内容', + `option_f` text COMMENT 'F选项内容', + `answer` varchar(32) DEFAULT NULL COMMENT '正确答案', + `answer_analysis` text COMMENT '答案解析', + `is_enable` int DEFAULT '1' COMMENT '是否禁用:1:启用,0:禁用,默认是1启用', + `is_del` int DEFAULT '0' COMMENT '是否删除:0未删除,1删除,默认0未删除', + `create_user` varchar(32) DEFAULT NULL COMMENT '创建人', + `create_time` datetime DEFAULT NULL COMMENT '创建时间', + `modify_user` varchar(32) DEFAULT NULL COMMENT '修改人', + `modify_time` datetime DEFAULT NULL COMMENT '修改时间,用于排序,创建时,修改时间等于创建时间', + PRIMARY KEY (`id`) +) ENGINE=InnoDB AUTO_INCREMENT=53 DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci COMMENT='记录试题信息'; + +-- ---------------------------- +-- Table structure for tms_xlsx_template +-- ---------------------------- +DROP TABLE IF EXISTS `tms_xlsx_template`; +CREATE TABLE `tms_xlsx_template` ( + `id` int NOT NULL AUTO_INCREMENT COMMENT '主键', + `file_name` varchar(300) DEFAULT NULL COMMENT '文件名', + `file_url` varchar(500) DEFAULT NULL COMMENT '文件全路径', + `status` int DEFAULT NULL COMMENT '状态', + PRIMARY KEY (`id`) +) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci COMMENT='模板文件信息记录'; + +-- ---------------------------- +-- Records of tms_xlsx_template +-- ---------------------------- +INSERT INTO `tms_xlsx_template` VALUES ('1', 'xlsx/20200826/1598430904075.xlsx', 'http://liuwanr.oss-cn-shenzhen.aliyuncs.com/xlsx/20200826/1598430904075.xlsx', null); + +-- ---------------------------- +-- Table structure for tms_evaluation_question +-- ---------------------------- +DROP TABLE IF EXISTS `tms_evaluation_question`; +CREATE TABLE `tms_evaluation_question` ( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自增主键', + `evaluation_record_id` int(11) DEFAULT NULL COMMENT '测评记录表id', + `question_id` int(11) DEFAULT NULL COMMENT '试题id', + `question_sort_no` int(11) DEFAULT NULL COMMENT '试题顺序号', + `question_status` int(11) DEFAULT '0' COMMENT '试题状态:是否已被作过:1、已作/0、未作', + `user_answer` varchar(32) DEFAULT NULL COMMENT '用户填写的答案', + `is_ture` int(11) DEFAULT '0' COMMENT '是否正确:1、正确/0、错误', + `question_points` int(11) DEFAULT NULL COMMENT '试题分值', + `question_score` int(11) DEFAULT NULL COMMENT '试题得分', + PRIMARY KEY (`id`) +) ENGINE=InnoDB AUTO_INCREMENT=21 DEFAULT CHARSET=utf8 COMMENT='测评试题表,记录每次测评的试题,与测评记录表相关联'; + +-- ---------------------------- +-- Table structure for tms_evaluation_record +-- ---------------------------- +DROP TABLE IF EXISTS `tms_evaluation_record`; +CREATE TABLE `tms_evaluation_record` ( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自增主键', + `user_id` int(11) DEFAULT NULL COMMENT '当前测评的用户的id', + `current_question_sort_no` int(11) DEFAULT '1' COMMENT '记录当前做到第几题,用于刷新页面以后保持在当前题目,默认开始时是第一题', + `total_question_num` int(11) DEFAULT NULL COMMENT '总题目数量', + `evaluation_status` int(11) DEFAULT '0' COMMENT '当前测评状态:1:已提交/0:未提交', + `start_time` datetime DEFAULT NULL COMMENT '开始时间', + `submit_time` datetime DEFAULT NULL COMMENT '提交时间', + PRIMARY KEY (`id`) +) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8 COMMENT='记录用户测评信息'; + + diff --git a/src/test/java/com/msdw/tms/IdWorker.java b/src/test/java/com/msdw/tms/IdWorker.java new file mode 100644 index 0000000..6c89e6f --- /dev/null +++ b/src/test/java/com/msdw/tms/IdWorker.java @@ -0,0 +1,146 @@ +package com.msdw.tms; + +/** + * @Author: gongsj. + * @Description: 自定义id生成器 + * @Date:Created in 2020/09/09 09:33. + * @Modified By: + */ +public class IdWorker { + + //因为二进制里第一个 bit 为如果是 1,那么都是负数,但是我们生成的 id 都是正数,所以第一个 bit 统一都是 0。 + + //机器ID 2进制5位 32位减掉1位 31个 + private long workerId; + //机房ID 2进制5位 32位减掉1位 31个 + private long datacenterId; + //代表一毫秒内生成的多个id的最新序号 12位 4096 -1 = 4095 个 + private long sequence; + //设置一个时间初始值 2^41 - 1 差不多可以用69年 + private long twepoch = 1585644268888L; + //5位的机器id + private long workerIdBits = 5L; + //5位的机房id + private long datacenterIdBits = 5L; + //每毫秒内产生的id数 2 的 12次方 + private long sequenceBits = 12L; + // 这个是二进制运算,就是5 bit最多只能有31个数字,也就是说机器id最多只能是32以内 + private long maxWorkerId = -1L ^ (-1L << workerIdBits); + // 这个是一个意思,就是5 bit最多只能有31个数字,机房id最多只能是32以内 + private long maxDatacenterId = -1L ^ (-1L << datacenterIdBits); + + private long workerIdShift = sequenceBits; + private long datacenterIdShift = sequenceBits + workerIdBits; + private long timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits; + private long sequenceMask = -1L ^ (-1L << sequenceBits); + //记录产生时间毫秒数,判断是否是同1毫秒 + private long lastTimestamp = -1L; + + public long getWorkerId() { + return workerId; + } + + public long getDatacenterId() { + return datacenterId; + } + + public long getTimestamp() { + return System.currentTimeMillis(); + } + + + public IdWorker(long workerId, long datacenterId, long sequence) { + + // 检查机房id和机器id是否超过31 不能小于0 + if (workerId > maxWorkerId || workerId < 0) { + throw new IllegalArgumentException( + String.format("worker Id can't be greater than %d or less than 0", maxWorkerId)); + } + + if (datacenterId > maxDatacenterId || datacenterId < 0) { + + throw new IllegalArgumentException( + String.format("datacenter Id can't be greater than %d or less than 0", maxDatacenterId)); + } + this.workerId = workerId; + this.datacenterId = datacenterId; + this.sequence = sequence; + } + + // 这个是核心方法,通过调用nextId()方法,让当前这台机器上的snowflake算法程序生成一个全局唯一的id + public synchronized long nextId() { + // 这儿就是获取当前时间戳,单位是毫秒 + long timestamp = timeGen(); + if (timestamp < lastTimestamp) { + + System.err.printf( + "clock is moving backwards. Rejecting requests until %d.", lastTimestamp); + throw new RuntimeException( + String.format("Clock moved backwards. Refusing to generate id for %d milliseconds", + lastTimestamp - timestamp)); + } + + // 下面是说假设在同一个毫秒内,又发送了一个请求生成一个id + // 这个时候就得把seqence序号给递增1,最多就是4096 + if (lastTimestamp == timestamp) { + + // 这个意思是说一个毫秒内最多只能有4096个数字,无论你传递多少进来, + //这个位运算保证始终就是在4096这个范围内,避免你自己传递个sequence超过了4096这个范围 + sequence = (sequence + 1) & sequenceMask; + //当某一毫秒的时间,产生的id数 超过4095,系统会进入等待,直到下一毫秒,系统继续产生ID + if (sequence == 0) { + timestamp = tilNextMillis(lastTimestamp); + } + + } else { + sequence = 0; + } + // 这儿记录一下最近一次生成id的时间戳,单位是毫秒 + lastTimestamp = timestamp; + // 这儿就是最核心的二进制位运算操作,生成一个64bit的id + // 先将当前时间戳左移,放到41 bit那儿;将机房id左移放到5 bit那儿;将机器id左移放到5 bit那儿;将序号放最后12 bit + // 最后拼接起来成一个64 bit的二进制数字,转换成10进制就是个long型 + return ((timestamp - twepoch) << timestampLeftShift) | + (datacenterId << datacenterIdShift) | + (workerId << workerIdShift) | sequence; + } + + /** + * 当某一毫秒的时间,产生的id数 超过4095,系统会进入等待,直到下一毫秒,系统继续产生ID + * + * @param lastTimestamp + * @return + */ + private long tilNextMillis(long lastTimestamp) { + + long timestamp = timeGen(); + + while (timestamp <= lastTimestamp) { + timestamp = timeGen(); + } + return timestamp; + } + + //获取当前时间戳 + private long timeGen() { + return System.currentTimeMillis(); + } + + /** + * main 测试类 + * + * @param args + */ + public static void main(String[] args) { +// System.out.println(1&4596); +// System.out.println(2&4596); +// System.out.println(6&4596); +// System.out.println(6&4596); +// System.out.println(6&4596); +// System.out.println(6&4596); + IdWorker worker = new IdWorker(1, 1, 1); + for (int i = 0; i < 22; i++) { + System.out.println(worker.nextId()); + } + } +} \ No newline at end of file diff --git a/src/test/java/com/msdw/tms/SnowflakeIdUtils.java b/src/test/java/com/msdw/tms/SnowflakeIdUtils.java new file mode 100644 index 0000000..cf43898 --- /dev/null +++ b/src/test/java/com/msdw/tms/SnowflakeIdUtils.java @@ -0,0 +1,174 @@ +package com.msdw.tms; + +/** + * @Author:JCccc + * @Description: + * @Date: created in 15:31 2019/6/12 + */ +public class SnowflakeIdUtils { + // ==============================Fields=========================================== + /** + * 开始时间截 (2015-01-01) + */ + private final long twepoch = 1420041600000L; + + /** + * 机器id所占的位数 + */ + private final long workerIdBits = 5L; + + /** + * 数据标识id所占的位数 + */ + private final long datacenterIdBits = 5L; + + /** + * 支持的最大机器id,结果是31 (这个移位算法可以很快的计算出几位二进制数所能表示的最大十进制数) + */ + private final long maxWorkerId = -1L ^ (-1L << workerIdBits); + + /** + * 支持的最大数据标识id,结果是31 + */ + private final long maxDatacenterId = -1L ^ (-1L << datacenterIdBits); + + /** + * 序列在id中占的位数 + */ + private final long sequenceBits = 12L; + + /** + * 机器ID向左移12位 + */ + private final long workerIdShift = sequenceBits; + + /** + * 数据标识id向左移17位(12+5) + */ + private final long datacenterIdShift = sequenceBits + workerIdBits; + + /** + * 时间截向左移22位(5+5+12) + */ + private final long timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits; + + /** + * 生成序列的掩码,这里为4095 (0b111111111111=0xfff=4095) + */ + private final long sequenceMask = -1L ^ (-1L << sequenceBits); + + /** + * 工作机器ID(0~31) + */ + private long workerId; + + /** + * 数据中心ID(0~31) + */ + private long datacenterId; + + /** + * 毫秒内序列(0~4095) + */ + private long sequence = 0L; + + /** + * 上次生成ID的时间截 + */ + private long lastTimestamp = -1L; + + //==============================Constructors===================================== + + /** + * 构造函数 + * + * @param workerId 工作ID (0~31) + * @param datacenterId 数据中心ID (0~31) + */ + public SnowflakeIdUtils(long workerId, long datacenterId) { + if (workerId > maxWorkerId || workerId < 0) { + throw new IllegalArgumentException(String.format("worker Id can't be greater than %d or less than 0", maxWorkerId)); + } + if (datacenterId > maxDatacenterId || datacenterId < 0) { + throw new IllegalArgumentException(String.format("datacenter Id can't be greater than %d or less than 0", maxDatacenterId)); + } + this.workerId = workerId; + this.datacenterId = datacenterId; + } + + // ==============================Methods========================================== + + /** + * 获得下一个ID (该方法是线程安全的) + * + * @return SnowflakeId + */ + public synchronized long nextId() { + long timestamp = timeGen(); + + //如果当前时间小于上一次ID生成的时间戳,说明系统时钟回退过这个时候应当抛出异常 + if (timestamp < lastTimestamp) { + throw new RuntimeException( + String.format("Clock moved backwards. Refusing to generate id for %d milliseconds", lastTimestamp - timestamp)); + } + + //如果是同一时间生成的,则进行毫秒内序列 + if (lastTimestamp == timestamp) { + sequence = (sequence + 1) & sequenceMask; + //毫秒内序列溢出 + if (sequence == 0) { + //阻塞到下一个毫秒,获得新的时间戳 + timestamp = tilNextMillis(lastTimestamp); + } + } + //时间戳改变,毫秒内序列重置 + else { + sequence = 0L; + } + + //上次生成ID的时间截 + lastTimestamp = timestamp; + + //移位并通过或运算拼到一起组成64位的ID + return ((timestamp - twepoch) << timestampLeftShift) // + | (datacenterId << datacenterIdShift) // + | (workerId << workerIdShift) // + | sequence; + } + + /** + * 阻塞到下一个毫秒,直到获得新的时间戳 + * + * @param lastTimestamp 上次生成ID的时间截 + * @return 当前时间戳 + */ + protected long tilNextMillis(long lastTimestamp) { + long timestamp = timeGen(); + while (timestamp <= lastTimestamp) { + timestamp = timeGen(); + } + return timestamp; + } + + /** + * 返回以毫秒为单位的当前时间 + * + * @return 当前时间(毫秒) + */ + protected long timeGen() { + return System.currentTimeMillis(); + } + + //==============================Test============================================= + + /** + * 测试 + */ + public static void main(String[] args) { + SnowflakeIdUtils idWorker = new SnowflakeIdUtils(3, 1); +// System.out.println(idWorker.nextId()); + for (int i = 0; i < 20; i++) { + System.out.println(idWorker.nextId()); + } + } +} \ No newline at end of file diff --git a/src/test/java/com/msdw/tms/TmsApplicationTests.java b/src/test/java/com/msdw/tms/TmsApplicationTests.java new file mode 100644 index 0000000..d936647 --- /dev/null +++ b/src/test/java/com/msdw/tms/TmsApplicationTests.java @@ -0,0 +1,518 @@ +package com.msdw.tms; + +import cn.hutool.core.date.DateUtil; +import com.baomidou.mybatisplus.core.metadata.IPage; +import com.baomidou.mybatisplus.extension.plugins.pagination.Page; +import com.msdw.tms.entity.request.QuestionsImportRequest; +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.data.redis.core.StringRedisTemplate; +import org.springframework.data.redis.core.ValueOperations; + +import java.io.UnsupportedEncodingException; +import java.net.URLDecoder; +import java.text.DecimalFormat; +import java.text.NumberFormat; +import java.text.ParseException; +import java.text.SimpleDateFormat; +import java.time.Instant; +import java.time.LocalDateTime; +import java.time.ZoneId; +import java.time.format.DateTimeFormatter; +import java.util.*; +import java.util.concurrent.TimeUnit; +import java.util.function.Function; +import java.util.stream.Collectors; + + +@SpringBootTest +class TmsApplicationTests { + + @Test + void contextLoads() throws UnsupportedEncodingException { + + System.out.println(URLDecoder.decode("%E8%AF%95%E9%A2%98%E8%A1%A8", "UTF-8")); + + //printMsg(1, 2, 3, "哈哈哈"); +// ------------------------------------------------------------------------------------------ + +// QuestionsVO questionsVO = new QuestionsVO(); +// +// QuestionsEntity questionsEntity = new QuestionsEntity(); +// +// questionsEntity.setId(1); +// questionsEntity.setQuestionType("单选题"); +// questionsEntity.setQuestionStem("adfasdfasdfasd"); +// questionsEntity.setQuestionTypeNo(1); +// questionsEntity.setIsEnable(1); +// questionsEntity.setIsDel(1); +// questionsEntity.setCreateUser("eqweq"); +// questionsEntity.setModifyUser("sadsa"); +// questionsEntity.setCreateTime(new Date()); +// questionsEntity.setModifyTime(new Date()); +// questionsEntity.setAnswer("asdasdasdsd"); +// questionsEntity.setAnswerAnalysis("qeqwewqeqweeqweqwe"); +// questionsEntity.setOptionA("dasdsada"); +// questionsEntity.setOptionB("asdasd"); +// questionsEntity.setOptionC("dasdsada"); +// questionsEntity.setOptionD("dasdsada"); +// questionsEntity.setOptionE("dasdsada"); +// questionsEntity.setOptionF("dasdsada"); +// questionsEntity.setSubjects("sadasd"); +// +// BeanUtils.copyProperties(questionsEntity, questionsVO); +// +// System.out.println(questionsVO.toString()); + +// ------------------------------------------------------------------------------------------ + +// Date date = new Date(); +// Instant instant = date.toInstant(); +// ZoneId zoneId = ZoneId.systemDefault(); +// +// LocalDateTime localDateTime = instant.atZone(zoneId).toLocalDateTime(); +// System.out.println("Date = " + date); +// System.out.println("LocalDateTime = " + localDateTime); +// String DateNow = localDateTime.format(DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss")); // 当前日期和时间 +// // String DateNow = LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss")); // 当前日期和时间 +// System.out.println(DateNow); + } + + void printMsg(T... args) { + for (T t : args) { + System.out.println("t is " + t); + } + } + + /** + * 泛型方法的基本介绍 + * + * @param tClass 传入的泛型实参 + * @return T 返回值为T类型 + * 说明: + * 1)public 与 返回值中间非常重要,可以理解为声明此方法为泛型方法。 + * 2)只有声明了的方法才是泛型方法,泛型类中的使用了泛型的成员方法并不是泛型方法。 + * 3)表明该方法将使用泛型类型T,此时才可以在方法中使用泛型类型T。 + * 4)与泛型类的定义一样,此处T可以随便写为任意标识,常见的如T、E、K、V等形式的参数常用于表示泛型。 + */ + public T genericMethod(Class tClass) throws InstantiationException, + IllegalAccessException { + T instance = tClass.newInstance(); + return instance; + } + + public K t2(T t, K k) { + k = (K) "haha"; + return k; + } + + private String formatDateTime(long mss) { + String DateTimes = null; + long days = mss / (60 * 60 * 24); + long hours = (mss % (60 * 60 * 24)) / (60 * 60); + long minutes = (mss % (60 * 60)) / 60; + long seconds = mss % 60; + if (days > 0) { + DateTimes = days + "天" + hours + "小时" + minutes + "分钟" + + seconds + "秒"; + } else if (hours > 0) { + DateTimes = hours + "小时" + minutes + "分钟" + + seconds + "秒"; + } else if (minutes > 0) { + DateTimes = minutes + "分钟" + + seconds + "秒"; + } else { + DateTimes = seconds + "秒"; + } + + return DateTimes; + } + + private String formatDateTime2(long mss) { + String DateTimes; + long hours = (mss % (60 * 24)) / (60); + long minutes = (mss % (60)); + //long seconds = mss % 60; + if (hours > 0) { + if (hours < 10) { + DateTimes = "0" + hours + ":" + minutes; + } else { + DateTimes = hours + ":" + minutes; + } + } else { + DateTimes = "00:" + minutes; + } + + return DateTimes; + } + + @Test + void t2() { + long mss = 150; + String ss = formatDateTime2(mss); + System.out.println(ss); + } + + /** + * 获取精确到秒的时间戳 + * + * @param date + * @return + */ + private int getSecondTimestampTwo(Date date) { + if (null == date) { + return 0; + } + String timestamp = String.valueOf(date.getTime() / 1000 / 60); + return Integer.valueOf(timestamp); + } + + @Test + void t3() { + String s = handleTime(new Date()); + System.out.println("ssss" + s); + + int secondTimestampTwo = getSecondTimestampTwo(new Date()); + System.out.println(secondTimestampTwo); + long l = System.currentTimeMillis(); + System.out.println("----" + new Date().getTime()); + System.out.println("====" + System.currentTimeMillis()); + + // System.currentTimeMillis() / (long)1000 / (long)60; + } + + @Test + void t4() throws ParseException { + Date date = getDate("2020-08-26 09:13:51"); + + int secondTimestampTwo = getSecondTimestampTwo(date); + + int secondTimestampTwo1 = getSecondTimestampTwo(new Date()); + + int i = secondTimestampTwo1 - secondTimestampTwo; + + System.out.println(i); + // System.currentTimeMillis() / (long)1000 / (long)60; + } + + //处理时间格式 + private String handleTime(Date date) { + Instant instant = date.toInstant(); + ZoneId zoneId = ZoneId.systemDefault(); + LocalDateTime localDateTime = instant.atZone(zoneId).toLocalDateTime(); + return localDateTime.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")); // 当前日期和时间 + } + + private Date getDate(String dateStr) throws ParseException { + SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");//注意月份是MM + Date date = simpleDateFormat.parse(dateStr); + System.out.println(date); //Mon Sep 02 00:00:00 CST 2019 + System.out.println(simpleDateFormat.format(date)); //2019-09-02 + return date; + } + + + @Autowired + StringRedisTemplate stringRedisTemplate; + + + @Test + void t5() throws ParseException { + + ValueOperations ops = stringRedisTemplate.opsForValue(); + + ops.set("aaa", "sss", 30, TimeUnit.SECONDS); + } + + @Test + void t6() throws ParseException { + + int se = 130 * 60; + + String s = DateUtil.secondToTime(se); + + System.out.println(s); + } + + @Test + void t7() throws ParseException { + + String f = "hahdhfasdf.xlsx"; + + System.out.println(f.substring(f.lastIndexOf("."))); + + } + + @Test + void t8() { + + List list = new ArrayList<>(); + list.add("a"); + list.add("b"); + list.add("c"); + list.add("d"); + list.add("b"); + list.add("c"); + list.add("a"); + list.add("a"); + list.add("a"); + + System.out.println("\n例子1 - 计算'a'出现的次数"); + System.out.println("a : " + Collections.frequency(list, "a")); + + System.out.println("\n例子2 - 计算所有对象出现的次数"); + Set uniqueSet = new HashSet<>(list); + for (String temp : uniqueSet) { + System.out.println(temp + ": " + Collections.frequency(list, temp)); + } + + System.out.println("\n例子3 -用Map来计算对象出现的次数"); + Map map = new HashMap<>(); + + for (String temp : list) { + Integer count = map.get(temp); + map.put(temp, (count == null) ? 1 : count + 1); + } + printMap(map); + + System.out.println("\nMap排序-以key排序"); + Map treeMap = new TreeMap<>(map); + printMap(treeMap); + + } + + static void printMap(Map map) { + for (Map.Entry entry : map.entrySet()) { + System.out.println("Key-value : " + entry.getKey() + "- " + + entry.getValue()); + } + } + + + @Test + void t9() throws Exception { + + List list = new ArrayList<>(); + list.add(new UserTest(1, "张三", "男", "深圳", "123456")); + list.add(new UserTest(2, "李四", "男", "深圳", "123456")); + list.add(new UserTest(3, "张三", "男", "杭州", "123456")); + list.add(new UserTest(4, "张三", "男", "深圳", "123456")); + list.add(new UserTest(5, "王五", "男", "深圳", "123456")); + list.add(new UserTest(6, "张三", "男", "深圳", "123456")); + list.add(new UserTest(7, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(8, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(9, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(10, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(11, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(12, "张三", "男", "菲律宾", "123456")); + list.add(new UserTest(13, "张三", "男", "深圳", "123456")); + list.add(new UserTest(14, "张三", "男", "马来西亚", "123456")); + + + List errorMsg = new ArrayList<>(); + + ArrayList collect = list.stream() + .collect(Collectors.collectingAndThen(// 去重收集 + Collectors.toCollection(() -> + new TreeSet<>(Comparator.comparing(UserTest::toString))), + ArrayList::new + )); + + for (UserTest userTest : collect) { + + System.out.println(userTest.getIndex() + "-----" + userTest.toString()); + } + + System.out.println("----------------------------"); + + List userTests = removeRepeatFactor3(collect, list); + + for (UserTest userTest : userTests) { + + System.out.println(userTest.getIndex() + "-----" + userTest.toString()); + } + + + } + + private List removeRepeatFactor(List list1, List list2) { + if (list1 != null && list2 != null) { + if (list1.size() != 0 && list2.size() != 0) { + Collection A = new ArrayList<>(list1); + List B = new ArrayList<>(list2); + A.retainAll(B); + if (A.size() != 0) { + B.removeAll(A); + } + return B; + } + } + return list2; + } + + private List removeRepeatFactor3(List list1, List list2) { + if (list1 != null && list2 != null) { + if (list1.size() != 0 && list2.size() != 0) { + Collection A = new ArrayList<>(list1); + List B = new ArrayList<>(list2); + A.retainAll(B); + if (A.size() != 0) { + B.removeAll(A); + } + return B; + } + } + return list2; + } + + public List removeRepeatFactor2(List list1, List list2) throws Exception { + if (list1 != null && list2 != null) { + if (list1.size() != 0 && list2.size() != 0) { + Collection A = new ArrayList<>(list1); + Collection B = new ArrayList<>(list2); + A.retainAll(B); + if (A.size() != 0) { + B.removeAll(A); + } + return (List) B; + } + } + return list2; + } + + + @Test + void t10() { + + QuestionsImportRequest questionsImportRequest = new QuestionsImportRequest(); + questionsImportRequest.setOptionA("aaa"); + questionsImportRequest.setOptionB("bbb"); + questionsImportRequest.setOptionC("ccc"); + questionsImportRequest.setOptionD("ddd"); + questionsImportRequest.setOptionE("eee"); + questionsImportRequest.setOptionF("fff"); + questionsImportRequest.setQuestionStem("stem"); + questionsImportRequest.setQuestionTypeName("type"); + + System.out.println(questionsImportRequest.toStringForCompare()); + + } + + @Test + void t11() { + + List list = new ArrayList<>(); + list.add(new UserTest(1, "张三", "男", "深圳", "123456")); + list.add(new UserTest(2, "李四", "男", "深圳", "123456")); + list.add(new UserTest(3, "张三", "男", "杭州", "123456")); + list.add(new UserTest(4, "张三", "男", "深圳", "123456")); + list.add(new UserTest(5, "王五", "男", "深圳", "123456")); + list.add(new UserTest(6, "张三", "男", "深圳", "123456")); + list.add(new UserTest(7, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(8, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(9, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(10, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(11, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(12, "张三", "男", "菲律宾", "123456")); + list.add(new UserTest(13, "张三", "男", "深圳", "123456")); + list.add(new UserTest(14, "张三", "男", "马来西亚", "123456")); + + stringRedisTemplate.opsForList().leftPush("sss", list.toString()); + } + + @Test + void t12() { +// System.out.println(((float) 3 / (float) 10) * 100 + "%"); + + int diliverNum = 3;//举例子的变量 + int queryMailNum = 10;//举例子的变量 +// 创建一个数值格式化对象 + NumberFormat numberFormat = NumberFormat.getInstance(); +// 设置精确到小数点后2位 + numberFormat.setMaximumFractionDigits(2); + String result = numberFormat.format((float) diliverNum / (float) queryMailNum * 100); + System.out.println("diliverNum和queryMailNum的百分比为:" + result + "%"); + } + + + public String getPercent(int x, int total) { + String result = "";//接受百分比的值 + double x_double = x * 1.0; + double tempresult = x / total; + //NumberFormat nf = NumberFormat.getPercentInstance(); 注释掉的也是一种方法 + //nf.setMinimumFractionDigits( 2 ); 保留到小数点后几位 + DecimalFormat df1 = new DecimalFormat("0.00%"); //##.00% 百分比格式,后面不足2位的用0补齐 + //result=nf.format(tempresult); + result = df1.format(tempresult); + return result; + } + + + @Test + public void test100() { + + List list = new ArrayList<>(); + list.add(new UserTest(1, "张三", "男", "深圳", "123456")); + list.add(new UserTest(2, "李四", "男", "深圳", "123456")); + list.add(new UserTest(3, "张三", "男", "杭州", "123456")); + list.add(new UserTest(4, "张三", "男", "深圳", "123456")); + list.add(new UserTest(5, "王五", "男", "深圳", "123456")); + list.add(new UserTest(6, "张三", "男", "深圳", "123456")); + list.add(new UserTest(7, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(8, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(9, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(10, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(11, "赵六", "女", "深圳", "123456")); + list.add(new UserTest(12, "张三", "男", "菲律宾", "123456")); + list.add(new UserTest(13, "张三", "男", "深圳", "123456")); + list.add(new UserTest(14, "张三", "男", "马来西亚", "123456")); + + Page page = new Page<>(1, 10); + +// page.setTotal(list.size()); +// page.setRecords(list); + + IPage convert = page.convert(new Function() { + @Override + public UserTest apply(Object o) { + + return new UserTest(1, "张三", "男", "深圳", "123456"); + } + }); + + System.out.println(convert); + + + } + + @Data + @NoArgsConstructor + @AllArgsConstructor + class T { + boolean b; + String s; + } + + @Test + public void t101() { + String[] s = new String[]{"5", "-2", "4", "C", "D", "9", "+", "+"}; + + List list = new ArrayList<>(); + + for (int i = 0; i < s.length; i++) { + + list.add(new T(true, s[i])); + + if (s[i].equalsIgnoreCase("c")) { + + } + } + + System.out.println(); + } + +} diff --git a/src/test/java/com/msdw/tms/UserTest.java b/src/test/java/com/msdw/tms/UserTest.java new file mode 100644 index 0000000..84e2982 --- /dev/null +++ b/src/test/java/com/msdw/tms/UserTest.java @@ -0,0 +1,26 @@ +package com.msdw.tms; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@NoArgsConstructor +@AllArgsConstructor +public class UserTest { + private int index; + private String name; + private String gender; + private String addr; + private String phone; + + @Override + public String toString() { + return "UserTest{" + + "name='" + name + '\'' + + ", gender='" + gender + '\'' + + ", addr='" + addr + '\'' + + ", phone='" + phone + '\'' + + '}'; + } +} diff --git a/src/test/java/com/msdw/tms/service/AliyunOssServiceTest.java b/src/test/java/com/msdw/tms/service/AliyunOssServiceTest.java new file mode 100644 index 0000000..c9583fd --- /dev/null +++ b/src/test/java/com/msdw/tms/service/AliyunOssServiceTest.java @@ -0,0 +1,62 @@ +package com.msdw.tms.service; + +import com.msdw.tms.common.utils.FilesResult; +import org.apache.commons.fileupload.FileItem; +import org.apache.commons.fileupload.disk.DiskFileItem; +import org.apache.commons.io.IOUtils; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.mock.web.MockHttpServletResponse; +import org.springframework.web.multipart.MultipartFile; +import org.springframework.web.multipart.commons.CommonsMultipartFile; + +import javax.servlet.http.HttpServletResponse; +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.nio.file.Files; + +@SpringBootTest +class AliyunOssServiceTest { + + @Autowired + AliyunOssService aliyunOssService; + + /** + * 测试上传文件 + */ + @Test + void testUploadFiles() throws IOException { + File file = new File("D:\\pic\\1.jpg"); + FileItem fileItem = new DiskFileItem("file", + Files.probeContentType(file.toPath()), + false, + file.getName(), + (int) file.length(), + file.getParentFile()); + IOUtils.copy(new FileInputStream(file), fileItem.getOutputStream()); + MultipartFile multipartFile = new CommonsMultipartFile(fileItem); + FilesResult filesResult = aliyunOssService.uploadFiles(multipartFile); + System.out.println(filesResult.toString()); + } + + /** + * 测试下载文件 + */ + @Test + void testDownloadFiles() throws IOException { + HttpServletResponse response = new MockHttpServletResponse(); + AliyunOssService aliyunOssService = this.aliyunOssService; + aliyunOssService.downloadFiles(response, + "http://liuwanr.oss-cn-shenzhen.aliyuncs.com/jpg/20200807/1596787474773.jpg"); + } + + /** + * 测试根据文件路径+文件名称,删除该文件 + */ + @Test + void testDeleteFile() { + aliyunOssService.deleteFile("http://liuwanr.oss-cn-shenzhen.aliyuncs.com/jpg/20200807/1596787474773.jpg"); + } +}