MintOrderService.java 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450
  1. package com.izouma.nineth.service;
  2. import com.alibaba.fastjson.JSON;
  3. import com.alibaba.fastjson.JSONObject;
  4. import com.alibaba.fastjson.serializer.SerializerFeature;
  5. import com.github.binarywang.wxpay.bean.order.WxPayMpOrderResult;
  6. import com.github.binarywang.wxpay.bean.order.WxPayMwebOrderResult;
  7. import com.github.binarywang.wxpay.bean.request.WxPayUnifiedOrderRequest;
  8. import com.github.binarywang.wxpay.constant.WxPayConstants;
  9. import com.github.binarywang.wxpay.exception.WxPayException;
  10. import com.github.binarywang.wxpay.service.WxPayService;
  11. import com.huifu.adapay.core.exception.BaseAdaPayException;
  12. import com.huifu.adapay.model.AdapayCommon;
  13. import com.huifu.adapay.model.Payment;
  14. import com.izouma.nineth.config.AdapayProperties;
  15. import com.izouma.nineth.config.GeneralProperties;
  16. import com.izouma.nineth.config.WxPayProperties;
  17. import com.izouma.nineth.domain.*;
  18. import com.izouma.nineth.dto.PageQuery;
  19. import com.izouma.nineth.enums.AssetStatus;
  20. import com.izouma.nineth.enums.MintOrderStatus;
  21. import com.izouma.nineth.enums.PayMethod;
  22. import com.izouma.nineth.exception.BusinessException;
  23. import com.izouma.nineth.repo.*;
  24. import com.izouma.nineth.utils.JpaUtils;
  25. import com.izouma.nineth.utils.SecurityUtils;
  26. import com.izouma.nineth.utils.SnowflakeIdWorker;
  27. import lombok.AllArgsConstructor;
  28. import lombok.extern.slf4j.Slf4j;
  29. import org.apache.commons.codec.EncoderException;
  30. import org.apache.commons.codec.net.URLCodec;
  31. import org.apache.commons.collections.MapUtils;
  32. import org.apache.commons.lang3.StringUtils;
  33. import org.springframework.core.env.Environment;
  34. import org.springframework.data.domain.Page;
  35. import org.springframework.scheduling.annotation.Scheduled;
  36. import org.springframework.stereotype.Service;
  37. import javax.transaction.Transactional;
  38. import java.math.BigDecimal;
  39. import java.math.RoundingMode;
  40. import java.time.LocalDateTime;
  41. import java.time.format.DateTimeFormatter;
  42. import java.util.*;
  43. import java.util.stream.Collectors;
  44. @Slf4j
  45. @Service
  46. @AllArgsConstructor
  47. public class MintOrderService {
  48. private MintOrderRepo mintOrderRepo;
  49. private UserRepo userRepo;
  50. private AssetService assetService;
  51. private AssetRepo assetRepo;
  52. private MintActivityRepo mintActivityRepo;
  53. private UserAddressRepo userAddressRepo;
  54. private GeneralProperties generalProperties;
  55. private Environment env;
  56. private AdapayProperties adapayProperties;
  57. private SnowflakeIdWorker snowflakeIdWorker;
  58. private WxPayProperties wxPayProperties;
  59. private WxPayService wxPayService;
  60. private MintMaterialRepo mintMaterialRepo;
  61. private MintActivityService mintActivityService;
  62. public Page<MintOrder> all(PageQuery pageQuery) {
  63. return mintOrderRepo.findAll(JpaUtils.toSpecification(pageQuery, MintOrder.class), JpaUtils.toPageRequest(pageQuery));
  64. }
  65. @Transactional
  66. public void create(Long userId, List<Long> assetIds) {
  67. User user = userRepo.findByIdAndDelFalse(userId).orElseThrow(new BusinessException("用户不存在"));
  68. User blackHole = userRepo.findByIdAndDelFalse(1435297L).orElseThrow(new BusinessException("无法铸造"));
  69. if (assetIds.size() != 3) {
  70. throw new BusinessException("数量不正确,请重新选择");
  71. }
  72. List<Asset> assets = assetRepo.findAllByIdInAndUserId(assetIds, userId);
  73. assets = assets.stream()
  74. .filter(asset -> asset.getName().contains("尼尔斯") && AssetStatus.NORMAL.equals(asset.getStatus()))
  75. .collect(Collectors.toList());
  76. if (assets.size() != 3) {
  77. throw new BusinessException("有藏品不符合,请重新选择");
  78. }
  79. // 铸造订单
  80. MintOrder order = mintOrderRepo.save(MintOrder.builder()
  81. .userId(userId)
  82. .phone(user.getPhone())
  83. // .material(materials)
  84. .consume(true)
  85. .status(MintOrderStatus.AIR_DROP)
  86. .build());
  87. // 铸造资产
  88. List<MintMaterial> materials = assets.stream().map(asset -> {
  89. MintMaterial material = new MintMaterial();
  90. material.setAssetId(asset.getId());
  91. material.setCollectionId(asset.getCollectionId());
  92. material.setName(asset.getName());
  93. material.setNumber(asset.getNumber());
  94. material.setPic(asset.getPic());
  95. material.setCategory(asset.getCategory());
  96. material.setOrderId(order.getId());
  97. return material;
  98. }).collect(Collectors.toList());
  99. mintMaterialRepo.saveAll(materials);
  100. // 改为转赠
  101. assets.forEach(asset -> assetService.transfer(asset, asset.getPrice(), blackHole, "转赠", null));
  102. }
  103. /**
  104. * 订单
  105. *
  106. * @param id 编号
  107. */
  108. public void finish(Long id) {
  109. MintOrder mintOrder = mintOrderRepo.findById(id).orElseThrow(new BusinessException("铸造订单不存在"));
  110. mintOrder.setStatus(MintOrderStatus.FINISH);
  111. mintOrderRepo.save(mintOrder);
  112. }
  113. /**
  114. * 发货
  115. *
  116. * @param id 编号
  117. * @param courierId 快递单号
  118. */
  119. public void dispatch(Long id, String courierId) {
  120. MintOrder mintOrder = mintOrderRepo.findById(id).orElseThrow(new BusinessException("铸造订单不存在"));
  121. mintOrder.setStatus(MintOrderStatus.RECEIVE);
  122. mintOrder.setCourierId(courierId);
  123. mintOrderRepo.save(mintOrder);
  124. }
  125. /**
  126. * @param user 用户
  127. * @param assetId 资产
  128. * @param mintActivityId 铸造活动
  129. * @param addressId 地址
  130. */
  131. @Transactional
  132. public MintOrder create(User user, List<Long> assetId, Long mintActivityId, Long addressId) {
  133. // 参加的活动
  134. MintActivity mintActivity = mintActivityRepo.findByIdAndDelFalse(mintActivityId)
  135. .orElseThrow(new BusinessException("无此铸造活动"));
  136. int stock = Optional.ofNullable(mintActivityService.decreaseStock(mintActivityId, 1))
  137. .map(Math::toIntExact)
  138. .orElseThrow(new BusinessException("很遗憾,藏品已售罄"));
  139. if (stock < 0) {
  140. throw new BusinessException("铸造活动已无库存");
  141. }
  142. if (mintActivity.getNum() > 0) {
  143. if (assetId.size() != mintActivity.getNum()) {
  144. throw new BusinessException("数量不正确,请重新选择");
  145. }
  146. }
  147. List<Asset> assets = assetRepo.findAllByIdInAndUserId(assetId, user.getId());
  148. // 资产产品是否符合铸造活动的名称
  149. assets = assets.stream()
  150. .filter(asset -> asset.getName()
  151. .contains(mintActivity.getCollectionName()) && AssetStatus.NORMAL.equals(asset.getStatus()))
  152. .collect(Collectors.toList());
  153. if (mintActivity.getNum() > 0 && (assets.size() != mintActivity.getNum())) {
  154. throw new BusinessException("有藏品不符合,请重新选择");
  155. }
  156. Map<Long, Long> privilegeIds = new HashMap<>();
  157. // 铸造特权
  158. if (!mintActivity.isConsume()) {
  159. assets.forEach(asset -> {
  160. List<Privilege> privileges = asset.getPrivileges()
  161. .stream()
  162. .filter(p -> p.getName().equals("铸造"))
  163. .collect(Collectors.toList());
  164. if (privileges.size() == 0) {
  165. throw new BusinessException("无铸造特权");
  166. } else {
  167. boolean flag = false;
  168. for (Privilege privilege : privileges) {
  169. // 打开多次 或者 可打开一次但未使用
  170. if (!privilege.isOnce() || (privilege.isOnce() && !privilege
  171. .isOpened())) {
  172. flag = true;
  173. privilegeIds.put(asset.getId(), privilege.getId());
  174. break;
  175. }
  176. }
  177. if (!flag) {
  178. throw new BusinessException("铸造特权已使用");
  179. }
  180. }
  181. });
  182. assets.forEach(asset -> {
  183. asset.getPrivileges()
  184. .stream()
  185. .filter(p -> p.getId().equals(privilegeIds.get(asset.getId())))
  186. .forEach(p -> {
  187. p.setOpened(true);
  188. p.setOpenTime(LocalDateTime.now());
  189. p.setOpenedBy(SecurityUtils.getAuthenticatedUser().getId());
  190. });
  191. assetRepo.save(asset);
  192. });
  193. } else {
  194. // 消耗改为转赠
  195. assets.forEach(asset -> {
  196. asset.setStatus(AssetStatus.MINTING);
  197. assetRepo.save(asset);
  198. });
  199. // 转让的用户
  200. userRepo.findByIdAndDelFalse(1435297L).orElseThrow(new BusinessException("无法铸造"));
  201. }
  202. UserAddress userAddress = null;
  203. if (addressId != null) {
  204. userAddress = userAddressRepo.findById(addressId).orElseThrow(new BusinessException("地址信息不存在"));
  205. }
  206. // 铸造订单
  207. MintOrder mintOrder = mintOrderRepo.save(MintOrder.builder()
  208. .userId(user.getId())
  209. .phone(user.getPhone())
  210. // .material(materials)
  211. .consume(mintActivity.isConsume())
  212. .status(MintOrderStatus.NOT_PAID)
  213. .airDrop(mintActivity.isAirDrop())
  214. .gasPrice(mintActivity.getGasPrice())
  215. .mintActivityId(mintActivityId)
  216. .contactName(Optional.ofNullable(userAddress).map(UserAddress::getName).orElse(null))
  217. .contactPhone(Optional.ofNullable(userAddress).map(UserAddress::getPhone).orElse(null))
  218. .address(Optional.ofNullable(userAddress).map(u ->
  219. u.getProvinceName() + " " + u.getCityName() + " " + u.getDistrictName() + " " + u.getAddress())
  220. .orElse(null))
  221. .build());
  222. // 铸造资产
  223. List<MintMaterial> materials = assets.stream().map(asset -> {
  224. MintMaterial material = new MintMaterial();
  225. material.setAssetId(asset.getId());
  226. material.setCollectionId(asset.getCollectionId());
  227. material.setName(asset.getName());
  228. material.setPrivilegeId(privilegeIds.get(asset.getId()));
  229. material.setNumber(asset.getNumber());
  230. material.setPic(asset.getPic());
  231. material.setCategory(asset.getCategory());
  232. material.setOrderId(mintOrder.getId());
  233. return material;
  234. }).collect(Collectors.toList());
  235. mintMaterialRepo.saveAll(materials);
  236. //库存
  237. // mintActivity.setStock(mintActivity.getStock() - 1);
  238. // mintActivityRepo.save(mintActivity);
  239. mintActivityService.increaseSale(mintActivityId, 1);
  240. return mintOrder;
  241. }
  242. public Object payOrderWeixin(Long id, String tradeType, String openId) throws WxPayException, EncoderException {
  243. MintOrder order = mintOrderRepo.findById(id).orElseThrow(new BusinessException("订单不存在"));
  244. if (order.getStatus() != MintOrderStatus.NOT_PAID) {
  245. throw new BusinessException("订单状态错误");
  246. }
  247. WxPayUnifiedOrderRequest request = new WxPayUnifiedOrderRequest();
  248. request.setBody("铸造GAS费");
  249. request.setOutTradeNo(String.valueOf(new SnowflakeIdWorker(1, 1).nextId()));
  250. request.setTotalFee(order.getGasPrice().multiply(BigDecimal.valueOf(100)).intValue());
  251. if (Arrays.stream(env.getActiveProfiles()).noneMatch(s -> s.equals("prod"))) {
  252. // 测试环境设为1分
  253. // request.setTotalFee(1);
  254. }
  255. request.setSpbillCreateIp("180.102.110.170");
  256. request.setNotifyUrl(wxPayProperties.getNotifyUrl());
  257. request.setTradeType(tradeType);
  258. request.setOpenid(openId);
  259. request.setSignType("MD5");
  260. JSONObject body = new JSONObject();
  261. body.put("action", "payMintOrder");
  262. body.put("userId", order.getUserId());
  263. body.put("orderId", order.getId());
  264. request.setAttach(body.toJSONString());
  265. if (WxPayConstants.TradeType.MWEB.equals(tradeType)) {
  266. WxPayMwebOrderResult result = wxPayService.createOrder(request);
  267. return result.getMwebUrl() + "&redirect_url=" + new URLCodec().encode(wxPayProperties.getReturnUrl());
  268. } else if (WxPayConstants.TradeType.JSAPI.equals(tradeType)) {
  269. return wxPayService.<WxPayMpOrderResult>createOrder(request);
  270. }
  271. throw new BusinessException("不支持此付款方式");
  272. }
  273. public Object payAdapay(Long id, String payChannel, String openId) throws BaseAdaPayException {
  274. List<String> aliChannels = Arrays.asList("alipay", "alipay_qr", "alipay_wap");
  275. List<String> wxChannels = Arrays.asList("wx_pub", "wx_lite");
  276. if (!aliChannels.contains(payChannel) && !wxChannels.contains(payChannel)) {
  277. throw new BusinessException("不支持此渠道");
  278. }
  279. MintOrder order = mintOrderRepo.findById(id).orElseThrow(new BusinessException("订单不存在"));
  280. if (order.getStatus() != MintOrderStatus.NOT_PAID) {
  281. throw new BusinessException("订单状态错误");
  282. }
  283. Map<String, Object> paymentParams = new HashMap<>();
  284. paymentParams.put("order_no", String.valueOf(snowflakeIdWorker.nextId()));
  285. paymentParams.put("pay_amt", order.getGasPrice().setScale(2, RoundingMode.HALF_UP).toPlainString());
  286. paymentParams.put("app_id", adapayProperties.getAppId());
  287. paymentParams.put("pay_channel", payChannel);
  288. paymentParams.put("goods_title", "铸造GAS费");
  289. paymentParams.put("goods_desc", "铸造GAS费");
  290. paymentParams.put("time_expire", DateTimeFormatter.ofPattern("yyyyMMddHHmmss")
  291. .format(LocalDateTime.now().plusMinutes(5)));
  292. paymentParams.put("notify_url", adapayProperties.getNotifyUrl() + "/mintOrder/" + order.getId());
  293. Map<String, Object> expend = new HashMap<>();
  294. paymentParams.put("expend", expend);
  295. if ("wx_pub".equals(payChannel)) {
  296. if (StringUtils.isBlank(openId)) {
  297. throw new BusinessException("缺少openId");
  298. }
  299. expend.put("open_id", openId);
  300. expend.put("limit_pay", "1");
  301. }
  302. Map<String, Object> response;
  303. if ("wx_lite".equals(payChannel)) {
  304. paymentParams.put("adapay_func_code", "wxpay.createOrder");
  305. paymentParams.put("callback_url", generalProperties.getHost() + "/9th/orders");
  306. response = AdapayCommon.requestAdapayUits(paymentParams);
  307. log.info("createOrderResponse {}", JSON.toJSONString(response, SerializerFeature.PrettyFormat));
  308. } else {
  309. response = Payment.create(paymentParams);
  310. log.info("createOrderResponse {}", JSON.toJSONString(response, SerializerFeature.PrettyFormat));
  311. AdapayService.checkSuccess(response);
  312. }
  313. switch (payChannel) {
  314. case "alipay_wap":
  315. case "alipay":
  316. return MapUtils.getString(MapUtils.getMap(response, "expend"), "pay_info");
  317. case "alipay_qr":
  318. return MapUtils.getString(MapUtils.getMap(response, "expend"), "qrcode_url");
  319. case "wx_pub":
  320. JSONObject payParams = JSON.parseObject(MapUtils.getString(MapUtils.getMap(response, "expend"), "pay_info"));
  321. payParams.put("timestamp", payParams.get("timeStamp"));
  322. payParams.remove("timeStamp");
  323. return payParams;
  324. default:
  325. return MapUtils.getMap(response, "expend");
  326. }
  327. }
  328. @Transactional
  329. public void mintNotify(Long orderId, PayMethod payMethod, String transactionId) {
  330. MintOrder mintOrder = mintOrderRepo.findById(orderId).orElseThrow(new BusinessException("订单不存在"));
  331. // List<MintMaterial> materials = mintOrder.getMaterial();
  332. List<MintMaterial> materials = mintMaterialRepo.findAllByOrderIdAndDelFalse(orderId);
  333. List<Asset> assets = assetRepo.findAllById(materials
  334. .stream()
  335. .map(MintMaterial::getAssetId)
  336. .collect(Collectors.toList()));
  337. mintOrder.setPayMethod(payMethod);
  338. if (mintOrder.isAirDrop()) {
  339. mintOrder.setStatus(MintOrderStatus.AIR_DROP);
  340. } else {
  341. mintOrder.setStatus(MintOrderStatus.DELIVERY);
  342. }
  343. mintOrder.setTransactionId(transactionId);
  344. mintOrder.setPayAt(LocalDateTime.now());
  345. if (mintOrder.isConsume()) {
  346. User newOwner = userRepo.findByIdAndDelFalse(1435297L).orElseThrow(new BusinessException("无法铸造"));
  347. assets.forEach(asset -> assetService.transfer(asset, asset.getPrice(), newOwner, "转赠", null));
  348. }
  349. mintOrderRepo.save(mintOrder);
  350. }
  351. @Scheduled(fixedRate = 60000)
  352. public void batchCancel() {
  353. if (generalProperties.isNotifyServer()) {
  354. return;
  355. }
  356. if (Arrays.asList(env.getActiveProfiles()).contains("dev")) {
  357. return;
  358. }
  359. List<MintOrder> orders = mintOrderRepo.findByStatusAndCreatedAtBeforeAndDelFalse(MintOrderStatus.NOT_PAID,
  360. LocalDateTime.now().minusMinutes(5));
  361. orders.forEach(o -> {
  362. try {
  363. cancel(o);
  364. } catch (Exception ignored) {
  365. }
  366. });
  367. }
  368. public void cancel(MintOrder order) {
  369. if (order.getStatus() != MintOrderStatus.NOT_PAID) {
  370. throw new BusinessException("已支付订单无法取消");
  371. }
  372. // List<MintMaterial> materials = order.getMaterial();
  373. List<MintMaterial> materials = mintMaterialRepo.findAllByOrderIdAndDelFalse(order.getId());
  374. List<Asset> assets = assetRepo.findAllById(materials.stream()
  375. .map(MintMaterial::getAssetId)
  376. .collect(Collectors.toList()));
  377. if (order.isConsume()) {
  378. assets.forEach(asset -> {
  379. asset.setStatus(AssetStatus.NORMAL);
  380. assetRepo.save(asset);
  381. });
  382. } else {
  383. Map<Long, Long> privilegeIds = materials.stream()
  384. .collect(Collectors.toMap(MintMaterial::getAssetId, MintMaterial::getPrivilegeId));
  385. assets.forEach(asset -> {
  386. asset.getPrivileges()
  387. .stream()
  388. .filter(p -> p.getId().equals(privilegeIds.get(asset.getId())))
  389. .forEach(p -> {
  390. p.setOpened(false);
  391. p.setOpenTime(null);
  392. p.setOpenedBy(null);
  393. });
  394. assetRepo.save(asset);
  395. });
  396. }
  397. log.info("set normal mintOrder {}", order.getId());
  398. order.setStatus(MintOrderStatus.CANCELLED);
  399. order.setCancelTime(LocalDateTime.now());
  400. mintOrderRepo.save(order);
  401. // 加库存
  402. // mintActivityRepo.addStock(order.getMintActivityId());
  403. mintActivityService.increaseStock(order.getMintActivityId(), 1);
  404. }
  405. }