Compare commits
3 Commits
cursor/gen
...
cursor/che
Author | SHA1 | Date | |
---|---|---|---|
![]() |
c796623148 | ||
![]() |
e6fecd8efe | ||
![]() |
3b2a3dd0ea |
@@ -10,7 +10,9 @@
|
||||
<if test="endTime != null">
|
||||
AND in_time < #{endTime}
|
||||
</if>
|
||||
AND tenant_id = ${@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getRequiredTenantId()}
|
||||
<if test="@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getTenantId() != null">
|
||||
AND tenant_id = ${@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getTenantId()}
|
||||
</if>
|
||||
AND deleted = 0) -
|
||||
(SELECT IFNULL(SUM(total_price), 0)
|
||||
FROM erp_purchase_return
|
||||
@@ -18,7 +20,9 @@
|
||||
<if test="endTime != null">
|
||||
AND return_time < #{endTime}
|
||||
</if>
|
||||
AND tenant_id = ${@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getRequiredTenantId()}
|
||||
<if test="@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getTenantId() != null">
|
||||
AND tenant_id = ${@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getTenantId()}
|
||||
</if>
|
||||
AND deleted = 0)
|
||||
</select>
|
||||
|
||||
|
@@ -10,7 +10,9 @@
|
||||
<if test="endTime != null">
|
||||
AND out_time < #{endTime}
|
||||
</if>
|
||||
AND tenant_id = ${@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getRequiredTenantId()}
|
||||
<if test="@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getTenantId() != null">
|
||||
AND tenant_id = ${@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getTenantId()}
|
||||
</if>
|
||||
AND deleted = 0) -
|
||||
(SELECT IFNULL(SUM(total_price), 0)
|
||||
FROM erp_sale_return
|
||||
@@ -18,7 +20,9 @@
|
||||
<if test="endTime != null">
|
||||
AND return_time < #{endTime}
|
||||
</if>
|
||||
AND tenant_id = ${@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getRequiredTenantId()}
|
||||
<if test="@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getTenantId() != null">
|
||||
AND tenant_id = ${@cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder@getTenantId()}
|
||||
</if>
|
||||
AND deleted = 0)
|
||||
</select>
|
||||
|
||||
|
@@ -0,0 +1,155 @@
|
||||
package cn.iocoder.yudao.module.erp.service.statistics;
|
||||
|
||||
import cn.iocoder.yudao.framework.tenant.core.context.TenantContextHolder;
|
||||
import cn.iocoder.yudao.module.erp.dal.mysql.statistics.ErpPurchaseStatisticsMapper;
|
||||
import cn.iocoder.yudao.module.erp.dal.mysql.statistics.ErpSaleStatisticsMapper;
|
||||
import org.junit.jupiter.api.AfterEach;
|
||||
import org.junit.jupiter.api.BeforeEach;
|
||||
import org.junit.jupiter.api.Test;
|
||||
import org.springframework.boot.test.context.SpringBootTest;
|
||||
import org.springframework.boot.test.mock.mockito.MockBean;
|
||||
import org.springframework.test.context.ActiveProfiles;
|
||||
|
||||
import jakarta.annotation.Resource;
|
||||
import java.math.BigDecimal;
|
||||
import java.time.LocalDateTime;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.*;
|
||||
import static org.mockito.ArgumentMatchers.any;
|
||||
import static org.mockito.Mockito.when;
|
||||
|
||||
/**
|
||||
* ERP 统计服务测试类
|
||||
* 主要测试在多租户关闭情况下,统计查询是否能正常工作
|
||||
*
|
||||
* @author 芋道源码
|
||||
*/
|
||||
@SpringBootTest
|
||||
@ActiveProfiles("unit-test")
|
||||
public class ErpStatisticsServiceTest {
|
||||
|
||||
@Resource
|
||||
private ErpSaleStatisticsService saleStatisticsService;
|
||||
|
||||
@Resource
|
||||
private ErpPurchaseStatisticsService purchaseStatisticsService;
|
||||
|
||||
@MockBean
|
||||
private ErpSaleStatisticsMapper saleStatisticsMapper;
|
||||
|
||||
@MockBean
|
||||
private ErpPurchaseStatisticsMapper purchaseStatisticsMapper;
|
||||
|
||||
@BeforeEach
|
||||
void setUp() {
|
||||
// 清理租户上下文
|
||||
TenantContextHolder.clear();
|
||||
}
|
||||
|
||||
@AfterEach
|
||||
void tearDown() {
|
||||
// 清理租户上下文
|
||||
TenantContextHolder.clear();
|
||||
}
|
||||
|
||||
@Test
|
||||
void testSaleStatisticsWithoutTenant() {
|
||||
// 准备参数
|
||||
LocalDateTime beginTime = LocalDateTime.of(2024, 1, 1, 0, 0, 0);
|
||||
LocalDateTime endTime = LocalDateTime.of(2024, 1, 31, 23, 59, 59);
|
||||
BigDecimal expectedPrice = new BigDecimal("1000.00");
|
||||
|
||||
// Mock 返回值
|
||||
when(saleStatisticsMapper.getSalePrice(any(LocalDateTime.class), any(LocalDateTime.class)))
|
||||
.thenReturn(expectedPrice);
|
||||
|
||||
// 测试:在没有租户ID的情况下调用销售统计
|
||||
assertDoesNotThrow(() -> {
|
||||
BigDecimal result = saleStatisticsService.getSalePrice(beginTime, endTime);
|
||||
assertEquals(expectedPrice, result);
|
||||
}, "在多租户关闭时,销售统计查询应该能正常工作");
|
||||
}
|
||||
|
||||
@Test
|
||||
void testPurchaseStatisticsWithoutTenant() {
|
||||
// 准备参数
|
||||
LocalDateTime beginTime = LocalDateTime.of(2024, 1, 1, 0, 0, 0);
|
||||
LocalDateTime endTime = LocalDateTime.of(2024, 1, 31, 23, 59, 59);
|
||||
BigDecimal expectedPrice = new BigDecimal("800.00");
|
||||
|
||||
// Mock 返回值
|
||||
when(purchaseStatisticsMapper.getPurchasePrice(any(LocalDateTime.class), any(LocalDateTime.class)))
|
||||
.thenReturn(expectedPrice);
|
||||
|
||||
// 测试:在没有租户ID的情况下调用采购统计
|
||||
assertDoesNotThrow(() -> {
|
||||
BigDecimal result = purchaseStatisticsService.getPurchasePrice(beginTime, endTime);
|
||||
assertEquals(expectedPrice, result);
|
||||
}, "在多租户关闭时,采购统计查询应该能正常工作");
|
||||
}
|
||||
|
||||
@Test
|
||||
void testSaleStatisticsWithTenant() {
|
||||
// 设置租户ID
|
||||
Long tenantId = 1L;
|
||||
TenantContextHolder.setTenantId(tenantId);
|
||||
|
||||
// 准备参数
|
||||
LocalDateTime beginTime = LocalDateTime.of(2024, 1, 1, 0, 0, 0);
|
||||
LocalDateTime endTime = LocalDateTime.of(2024, 1, 31, 23, 59, 59);
|
||||
BigDecimal expectedPrice = new BigDecimal("1500.00");
|
||||
|
||||
// Mock 返回值
|
||||
when(saleStatisticsMapper.getSalePrice(any(LocalDateTime.class), any(LocalDateTime.class)))
|
||||
.thenReturn(expectedPrice);
|
||||
|
||||
// 测试:在有租户ID的情况下调用销售统计
|
||||
assertDoesNotThrow(() -> {
|
||||
BigDecimal result = saleStatisticsService.getSalePrice(beginTime, endTime);
|
||||
assertEquals(expectedPrice, result);
|
||||
}, "在多租户开启时,销售统计查询应该能正常工作");
|
||||
|
||||
// 验证租户ID是否正确设置
|
||||
assertEquals(tenantId, TenantContextHolder.getTenantId());
|
||||
}
|
||||
|
||||
@Test
|
||||
void testPurchaseStatisticsWithTenant() {
|
||||
// 设置租户ID
|
||||
Long tenantId = 2L;
|
||||
TenantContextHolder.setTenantId(tenantId);
|
||||
|
||||
// 准备参数
|
||||
LocalDateTime beginTime = LocalDateTime.of(2024, 1, 1, 0, 0, 0);
|
||||
LocalDateTime endTime = LocalDateTime.of(2024, 1, 31, 23, 59, 59);
|
||||
BigDecimal expectedPrice = new BigDecimal("1200.00");
|
||||
|
||||
// Mock 返回值
|
||||
when(purchaseStatisticsMapper.getPurchasePrice(any(LocalDateTime.class), any(LocalDateTime.class)))
|
||||
.thenReturn(expectedPrice);
|
||||
|
||||
// 测试:在有租户ID的情况下调用采购统计
|
||||
assertDoesNotThrow(() -> {
|
||||
BigDecimal result = purchaseStatisticsService.getPurchasePrice(beginTime, endTime);
|
||||
assertEquals(expectedPrice, result);
|
||||
}, "在多租户开启时,采购统计查询应该能正常工作");
|
||||
|
||||
// 验证租户ID是否正确设置
|
||||
assertEquals(tenantId, TenantContextHolder.getTenantId());
|
||||
}
|
||||
|
||||
@Test
|
||||
void testTenantContextHolderMethods() {
|
||||
// 测试 getTenantId() 在没有设置租户时返回 null
|
||||
assertNull(TenantContextHolder.getTenantId(), "未设置租户时应该返回 null");
|
||||
|
||||
// 设置租户ID
|
||||
Long tenantId = 3L;
|
||||
TenantContextHolder.setTenantId(tenantId);
|
||||
assertEquals(tenantId, TenantContextHolder.getTenantId(), "设置租户后应该能正确获取");
|
||||
|
||||
// 清理租户上下文
|
||||
TenantContextHolder.clear();
|
||||
assertNull(TenantContextHolder.getTenantId(), "清理后应该返回 null");
|
||||
}
|
||||
}
|
@@ -40,10 +40,18 @@ public interface CouponTemplateMapper extends BaseMapperX<CouponTemplateDO> {
|
||||
.orderByDesc(CouponTemplateDO::getId));
|
||||
}
|
||||
|
||||
default void updateTakeCount(Long id, Integer incrCount) {
|
||||
update(null, new LambdaUpdateWrapper<CouponTemplateDO>()
|
||||
.eq(CouponTemplateDO::getId, id)
|
||||
.setSql("take_count = take_count + " + incrCount));
|
||||
default int updateTakeCount(Long id, Integer incrCount) {
|
||||
LambdaUpdateWrapper<CouponTemplateDO> wrapper = new LambdaUpdateWrapper<CouponTemplateDO>()
|
||||
.eq(CouponTemplateDO::getId, id);
|
||||
|
||||
// 只在增加数量时检查库存(incrCount > 0)
|
||||
if (incrCount > 0) {
|
||||
// 添加库存判断:剩余数量 >= 领取的数量,或者总数量为-1(无限库存)
|
||||
wrapper.and(w -> w.apply("total_count = -1 OR (total_count - take_count) >= {0}", incrCount));
|
||||
}
|
||||
|
||||
wrapper.setSql("take_count = take_count + " + incrCount);
|
||||
return update(null, wrapper);
|
||||
}
|
||||
|
||||
default List<CouponTemplateDO> selectListByTakeType(Integer takeType) {
|
||||
|
@@ -279,12 +279,8 @@ public class CouponServiceImpl implements CouponService {
|
||||
if (ObjUtil.notEqual(couponTemplate.getTakeType(), takeType.getType())) {
|
||||
throw exception(COUPON_TEMPLATE_CANNOT_TAKE);
|
||||
}
|
||||
// 校验发放数量不能过小(仅在 CouponTakeTypeEnum.USER 用户领取时)
|
||||
if (CouponTakeTypeEnum.isUser(couponTemplate.getTakeType())
|
||||
&& ObjUtil.notEqual(couponTemplate.getTakeLimitCount(), CouponTemplateDO.TIME_LIMIT_COUNT_MAX) // 非不限制
|
||||
&& couponTemplate.getTakeCount() + userIds.size() > couponTemplate.getTotalCount()) {
|
||||
throw exception(COUPON_TEMPLATE_NOT_ENOUGH);
|
||||
}
|
||||
// 注意:库存检查现在在数据库层面的 updateCouponTemplateTakeCount 方法中进行
|
||||
// 如果库存不足,该方法会抛出 COUPON_TEMPLATE_NOT_ENOUGH 异常
|
||||
// 校验"固定日期"的有效期类型是否过期
|
||||
if (CouponTemplateValidityTypeEnum.DATE.getType().equals(couponTemplate.getValidityType())) {
|
||||
if (LocalDateTimeUtils.beforeNow(couponTemplate.getValidEndTime())) {
|
||||
|
@@ -23,6 +23,7 @@ import java.util.Objects;
|
||||
|
||||
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
|
||||
import static cn.iocoder.yudao.module.promotion.enums.ErrorCodeConstants.COUPON_TEMPLATE_NOT_EXISTS;
|
||||
import static cn.iocoder.yudao.module.promotion.enums.ErrorCodeConstants.COUPON_TEMPLATE_NOT_ENOUGH;
|
||||
import static cn.iocoder.yudao.module.promotion.enums.ErrorCodeConstants.COUPON_TEMPLATE_TOTAL_COUNT_TOO_SMALL;
|
||||
|
||||
/**
|
||||
@@ -116,7 +117,11 @@ public class CouponTemplateServiceImpl implements CouponTemplateService {
|
||||
|
||||
@Override
|
||||
public void updateCouponTemplateTakeCount(Long id, int incrCount) {
|
||||
couponTemplateMapper.updateTakeCount(id, incrCount);
|
||||
int updateCount = couponTemplateMapper.updateTakeCount(id, incrCount);
|
||||
// 只在增加数量且更新失败时,说明库存不足,抛出异常
|
||||
if (incrCount > 0 && updateCount == 0) {
|
||||
throw exception(COUPON_TEMPLATE_NOT_ENOUGH);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
|
Reference in New Issue
Block a user