| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337 |
- package com.izouma.nineth.service;
- import com.alibaba.fastjson.JSON;
- import com.izouma.nineth.domain.Collection;
- import com.izouma.nineth.domain.*;
- import com.izouma.nineth.dto.CollectionDTO;
- import com.izouma.nineth.dto.CreateBlindBox;
- import com.izouma.nineth.dto.PageQuery;
- import com.izouma.nineth.enums.CollectionSource;
- import com.izouma.nineth.enums.CollectionType;
- import com.izouma.nineth.exception.BusinessException;
- import com.izouma.nineth.repo.*;
- import com.izouma.nineth.utils.JpaUtils;
- import com.izouma.nineth.utils.SecurityUtils;
- import lombok.AllArgsConstructor;
- import org.apache.commons.collections.MapUtils;
- import org.apache.commons.lang3.RandomUtils;
- import org.apache.commons.lang3.Range;
- import org.apache.commons.lang3.StringUtils;
- import org.springframework.beans.BeanUtils;
- import org.springframework.data.domain.Page;
- import org.springframework.data.domain.PageImpl;
- import org.springframework.data.domain.PageRequest;
- import org.springframework.data.domain.Sort;
- import org.springframework.data.jpa.domain.Specification;
- import org.springframework.scheduling.TaskScheduler;
- import org.springframework.stereotype.Service;
- import javax.annotation.PostConstruct;
- import javax.persistence.criteria.Predicate;
- import javax.transaction.Transactional;
- import java.time.LocalDateTime;
- import java.time.ZoneId;
- import java.util.*;
- import java.util.concurrent.ScheduledFuture;
- import java.util.stream.Collectors;
- @Service
- @AllArgsConstructor
- public class CollectionService {
- private CollectionRepo collectionRepo;
- private LikeRepo likeRepo;
- private BlindBoxItemRepo blindBoxItemRepo;
- private AppointmentRepo appointmentRepo;
- private UserRepo userRepo;
- private TaskScheduler taskScheduler;
- private CacheService cacheService;
- private final Map<Long, ScheduledFuture<?>> tasks = new HashMap<>();
- @PostConstruct
- public void init() {
- List<Collection> collections = collectionRepo.findByScheduleSaleTrueAndOnShelfFalseAndStartTimeBeforeAndDelFalse(LocalDateTime.now());
- for (Collection collection : collections) {
- onShelfTask(collection);
- }
- }
- public Page<Collection> all(PageQuery pageQuery) {
- pageQuery.getQuery().put("del", false);
- String type = MapUtils.getString(pageQuery.getQuery(), "type", "DEFAULT");
- pageQuery.getQuery().remove("type");
- Specification<Collection> specification = JpaUtils.toSpecification(pageQuery, Collection.class);
- PageRequest pageRequest = JpaUtils.toPageRequest(pageQuery);
- if (pageRequest.getSort().stream().noneMatch(order -> order.getProperty().equals("createdAt"))) {
- pageRequest = PageRequest.of(pageRequest.getPageNumber(), pageQuery.getSize(),
- pageRequest.getSort().and(Sort.by("createdAt").descending()));
- }
- specification = specification.and((Specification<Collection>) (root, criteriaQuery, criteriaBuilder) -> {
- List<Predicate> and = new ArrayList<>();
- if (StringUtils.isNotEmpty(type) && !"all".equalsIgnoreCase(type)) {
- try {
- if (type.contains(",")) {
- and.add(root.get("type")
- .in(Arrays.stream(type.split(",")).map(s -> Enum.valueOf(CollectionType.class, s))
- .collect(Collectors.toList())));
- } else {
- and.add(criteriaBuilder.equal(root.get("type"), Enum.valueOf(CollectionType.class, type)));
- }
- } catch (Exception e) {
- }
- }
- return criteriaBuilder.and(and.toArray(new Predicate[0]));
- });
- return collectionRepo.findAll(specification, pageRequest);
- }
- public Collection create(Collection record) {
- User minter = userRepo.findById(record.getMinterId()).orElse(SecurityUtils.getAuthenticatedUser());
- record.setMinter(minter.getNickname());
- record.setMinterId(minter.getId());
- record.setMinterAvatar(minter.getAvatar());
- record.setOwner(minter.getNickname());
- record.setOwnerId(minter.getId());
- record.setOwnerAvatar(minter.getAvatar());
- record.setStock(record.getTotal());
- record.setSale(0);
- if (record.isScheduleSale()) {
- if (record.getStartTime() == null) {
- throw new BusinessException("请填写定时发布时间");
- }
- if (record.getStartTime().isBefore(LocalDateTime.now())) {
- record.setOnShelf(true);
- record.setSalable(true);
- record.setStartTime(null);
- }
- }
- record = collectionRepo.save(record);
- onShelfTask(record);
- return record;
- }
- public Collection update(Collection record) {
- collectionRepo.update(record.getId(), record.isOnShelf(), record.isSalable(),
- record.getStartTime(), record.isScheduleSale(), record.getSort(),
- record.getDetail(), JSON.toJSONString(record.getPrivileges()),
- JSON.toJSONString(record.getProperties()), JSON.toJSONString(record.getModel3d()));
- record = collectionRepo.findById(record.getId()).orElseThrow(new BusinessException("无记录"));
- onShelfTask(record);
- return record;
- }
- private void onShelfTask(Collection record) {
- ScheduledFuture<?> task = tasks.get(record.getId());
- if (task != null) {
- if (!task.cancel(true)) {
- return;
- }
- }
- if (record.isScheduleSale()) {
- if (record.getStartTime().minusSeconds(2).isAfter(LocalDateTime.now())) {
- Date date = Date.from(record.getStartTime().atZone(ZoneId.systemDefault()).toInstant());
- ScheduledFuture<?> future = taskScheduler.schedule(() -> {
- collectionRepo.scheduleOnShelf(record.getId());
- tasks.remove(record.getId());
- }, date);
- tasks.put(record.getId(), future);
- } else {
- collectionRepo.scheduleOnShelf(record.getId());
- }
- }
- }
- public CollectionDTO toDTO(Collection collection) {
- return toDTO(collection, true);
- }
- public CollectionDTO toDTO(Collection collection, boolean join) {
- CollectionDTO collectionDTO = new CollectionDTO();
- BeanUtils.copyProperties(collection, collectionDTO);
- if (join) {
- if (SecurityUtils.getAuthenticatedUser() != null) {
- List<Like> list = likeRepo.findByUserIdAndCollectionId(SecurityUtils.getAuthenticatedUser().getId(),
- collection.getId());
- collectionDTO.setLiked(!list.isEmpty());
- if (collection.getType() == CollectionType.BLIND_BOX) {
- collectionDTO.setAppointment(appointmentRepo.findFirstByBlindBoxId(collection.getId()).isPresent());
- }
- }
- }
- return collectionDTO;
- }
- public List<CollectionDTO> toDTO(List<Collection> collections) {
- List<Like> likes = new ArrayList<>();
- List<Appointment> appointments = new ArrayList<>();
- if (SecurityUtils.getAuthenticatedUser() != null) {
- likes.addAll(likeRepo.findByUserId(SecurityUtils.getAuthenticatedUser().getId()));
- appointments.addAll(appointmentRepo.findByUserId(SecurityUtils.getAuthenticatedUser().getId()));
- }
- return collections.stream().parallel().map(collection -> {
- CollectionDTO dto = toDTO(collection, false);
- if (!likes.isEmpty()) {
- dto.setLiked(likes.stream().anyMatch(l -> l.getCollectionId().equals(collection.getId())));
- }
- if (!appointments.isEmpty()) {
- dto.setAppointment(appointments.stream().anyMatch(a -> a.getBlindBoxId().equals(collection.getId())));
- }
- return dto;
- }).collect(Collectors.toList());
- }
- public Page<CollectionDTO> toDTO(Page<Collection> collections) {
- List<CollectionDTO> userDTOS = toDTO(collections.getContent());
- return new PageImpl<>(userDTOS, collections.getPageable(), collections.getTotalElements());
- }
- @Transactional
- public Collection createBlindBox(CreateBlindBox createBlindBox) {
- Collection blindBox = createBlindBox.getBlindBox();
- if (blindBox.getId() != null) {
- throw new BusinessException("无法完成此操作");
- }
- List<Collection> list =
- collectionRepo.findAllById(createBlindBox.getItems().stream().map(BlindBoxItem::getCollectionId)
- .collect(Collectors.toSet()));
- for (BlindBoxItem item : createBlindBox.getItems()) {
- Collection collection = list.stream().filter(i -> i.getId().equals(item.getCollectionId())).findAny()
- .orElseThrow(new BusinessException("所选藏品不存在"));
- if (item.getTotal() > collection.getStock()) {
- throw new BusinessException("所选藏品库存不足:" + collection.getName());
- }
- }
- User user = userRepo.findById(blindBox.getMinterId()).orElse(SecurityUtils.getAuthenticatedUser());
- blindBox.setMinter(user.getNickname());
- blindBox.setMinterId(user.getId());
- blindBox.setMinterAvatar(user.getAvatar());
- blindBox.setOwner(user.getNickname());
- blindBox.setOwnerId(user.getId());
- blindBox.setOwnerAvatar(user.getAvatar());
- blindBox.setStock(blindBox.getTotal());
- blindBox.setSale(0);
- collectionRepo.save(blindBox);
- for (BlindBoxItem item : createBlindBox.getItems()) {
- Collection collection = list.stream().filter(i -> i.getId().equals(item.getCollectionId())).findAny()
- .orElseThrow(new BusinessException("所选藏品不存在"));
- collection.setStock(collection.getStock() - item.getTotal());
- collectionRepo.save(collection);
- BlindBoxItem blindBoxItem = new BlindBoxItem();
- BeanUtils.copyProperties(collection, blindBoxItem);
- blindBoxItem.setId(null);
- blindBoxItem.setCollectionId(item.getCollectionId());
- blindBoxItem.setSale(0);
- blindBoxItem.setTotal(item.getTotal());
- blindBoxItem.setStock(item.getTotal());
- blindBoxItem.setRare(item.isRare());
- blindBoxItem.setBlindBoxId(blindBox.getId());
- blindBoxItemRepo.save(blindBoxItem);
- }
- return blindBox;
- }
- public void appointment(Long id, Long userId) {
- Collection collection = collectionRepo.findById(id).orElseThrow(new BusinessException("无记录"));
- if (collection.getType() != CollectionType.BLIND_BOX) {
- throw new BusinessException("非盲盒,无需预约");
- }
- if (collection.getStartTime().isBefore(LocalDateTime.now())) {
- throw new BusinessException("盲盒已开售,无需预约");
- }
- appointmentRepo.save(Appointment.builder()
- .userId(userId)
- .blindBoxId(id)
- .build());
- }
- public BlindBoxItem draw(Long collectionId) {
- List<BlindBoxItem> items = blindBoxItemRepo.findByBlindBoxId(collectionId);
- Map<BlindBoxItem, Range<Integer>> randomRange = new HashMap<>();
- int c = 0, sum = 0;
- for (BlindBoxItem item : items) {
- randomRange.put(item, Range.between(c, c + item.getStock()));
- c += item.getStock();
- sum += item.getStock();
- }
- int retry = 0;
- BlindBoxItem winItem = null;
- while (winItem == null) {
- retry++;
- int rand = RandomUtils.nextInt(0, sum + 1);
- for (Map.Entry<BlindBoxItem, Range<Integer>> entry : randomRange.entrySet()) {
- BlindBoxItem item = entry.getKey();
- Range<Integer> range = entry.getValue();
- if (rand >= range.getMinimum() && rand < range.getMaximum()) {
- int total = items.stream().filter(i -> !i.isRare())
- .mapToInt(BlindBoxItem::getTotal).sum();
- int stock = items.stream().filter(i -> !i.isRare())
- .mapToInt(BlindBoxItem::getStock).sum();
- if (item.isRare()) {
- double nRate = stock / (double) total;
- double rRate = (item.getStock() - 1) / (double) item.getTotal();
- if (Math.abs(nRate - rRate) < (1 / (double) item.getTotal()) || retry > 1 || rRate == 0) {
- if (!(nRate > 0.1 && item.getStock() == 1)) {
- winItem = item;
- }
- }
- } else {
- double nRate = (stock - 1) / (double) total;
- double rRate = item.getStock() / (double) item.getTotal();
- if (Math.abs(nRate - rRate) < 0.2 || retry > 1 || nRate == 0) {
- winItem = item;
- }
- }
- }
- }
- if (retry > 100 && winItem == null) {
- throw new BusinessException("盲盒抽卡失败");
- }
- }
- winItem.setStock(winItem.getStock() - 1);
- winItem.setSale(winItem.getSale() + 1);
- blindBoxItemRepo.save(winItem);
- return winItem;
- }
- public synchronized Integer getNextNumber(Long collectionId) {
- collectionRepo.increaseNumber(collectionId, 1);
- return collectionRepo.getCurrentNumber(collectionId).orElse(0);
- }
- public synchronized Integer getNextNumber(Collection collection) {
- if (collection == null) return 0;
- if (collection.getCurrentNumber() == null) {
- collection.setCurrentNumber(0);
- }
- collection.setCurrentNumber(collection.getCurrentNumber() + 1);
- collectionRepo.save(collection);
- return collection.getCurrentNumber();
- }
- public void addStock(Long id, int number) {
- Collection collection = collectionRepo.findById(id).orElseThrow(new BusinessException("无记录"));
- if (collection.getSource() != CollectionSource.OFFICIAL) {
- throw new BusinessException("用户转售无法增发");
- }
- if (collection.getType() == CollectionType.BLIND_BOX) {
- throw new BusinessException("盲盒无法增发");
- }
- collection.setStock(collection.getStock() + number);
- collection.setTotal(collection.getTotal() + number);
- }
- }
|