refactor(image): 重构水印处理逻辑以提高可维护性

- 移除 PhotoProcessContext 中的水印相关字段
- 新增 WatermarkConfig 类封装水印配置
- 修改 WatermarkStage 通过构造函数注入配置
- 调整 PrinterServiceImpl 中水印配置的传递方式
- 更新单元测试以适应新的配置注入方式
- 统一从配置对象读取水印参数而非上下文
- 优化日志记录与偏移量计算逻辑
This commit is contained in:
2025-11-26 14:56:37 +08:00
parent 90efc908c5
commit 333c4d3ca7
6 changed files with 113 additions and 47 deletions

View File

@@ -101,12 +101,6 @@ public class PhotoProcessContext {
private String resultUrl;
private IStorageAdapter storageAdapter;
private WatermarkInfo watermarkInfo;
private ImageWatermarkOperatorEnum watermarkType;
private String scenicText;
private String dateFormat;
private File qrcodeFile;
private Integer offsetLeft;
// ==================== 回调 ====================

View File

@@ -23,8 +23,6 @@ import java.io.File;
)
public class ConditionalRotateStage extends AbstractPipelineStage<PhotoProcessContext> {
private static final int OFFSET_LEFT_FOR_PORTRAIT = 40;
@Override
public String getName() {
return "ConditionalRotateStage";
@@ -62,9 +60,8 @@ public class ConditionalRotateStage extends AbstractPipelineStage<PhotoProcessCo
context.updateProcessedFile(rotatedFile);
context.setNeedRotation(true);
context.setOffsetLeft(OFFSET_LEFT_FOR_PORTRAIT);
log.info("图片已旋转{}度, offsetLeft={}", rotation, OFFSET_LEFT_FOR_PORTRAIT);
log.info("图片已旋转{}度", rotation);
return StageResult.success("已旋转" + rotation + "");
} catch (Exception e) {

View File

@@ -0,0 +1,37 @@
package com.ycwl.basic.image.pipeline.stages;
import com.ycwl.basic.image.watermark.enums.ImageWatermarkOperatorEnum;
import lombok.Builder;
import lombok.Getter;
import java.io.File;
/**
* 水印Stage配置
* 封装水印处理所需的所有配置参数
*/
@Getter
@Builder
public class WatermarkConfig {
/**
* 水印类型
*/
private final ImageWatermarkOperatorEnum watermarkType;
/**
* 景区文字
*/
private final String scenicText;
/**
* 日期格式
*/
@Builder.Default
private final String dateFormat = "yyyy.MM.dd";
/**
* 二维码文件
*/
private final File qrcodeFile;
}

View File

@@ -31,6 +31,19 @@ import java.util.List;
)
public class WatermarkStage extends AbstractPipelineStage<PhotoProcessContext> {
private static final int OFFSET_LEFT_FOR_PORTRAIT = 40;
private final WatermarkConfig config;
/**
* 构造函数
*
* @param config 水印配置
*/
public WatermarkStage(WatermarkConfig config) {
this.config = config;
}
@Override
public String getName() {
return "WatermarkStage";
@@ -43,12 +56,13 @@ public class WatermarkStage extends AbstractPipelineStage<PhotoProcessContext> {
@Override
protected StageResult doExecute(PhotoProcessContext context) {
ImageWatermarkOperatorEnum watermarkType = context.getWatermarkType();
if (watermarkType == null) {
if (config == null || config.getWatermarkType() == null) {
log.info("未配置水印类型,跳过水印处理");
return StageResult.skipped("未配置水印");
}
ImageWatermarkOperatorEnum watermarkType = config.getWatermarkType();
List<ImageWatermarkOperatorEnum> fallbackChain = buildFallbackChain(watermarkType);
log.debug("水印降级链: {}", fallbackChain);
@@ -138,30 +152,29 @@ public class WatermarkStage extends AbstractPipelineStage<PhotoProcessContext> {
info.setOriginalFile(originalFile);
info.setWatermarkedFile(watermarkedFile);
String scenicText = context.getScenicText();
// 从 config 读取景区文字
String scenicText = config.getScenicText();
if (StringUtils.isNotBlank(scenicText)) {
info.setScenicLine(scenicText);
}
// 从 config 读取日期格式
Date now = new Date();
String dateFormat = context.getDateFormat();
if (StringUtils.isBlank(dateFormat)) {
dateFormat = "yyyy.MM.dd";
}
String dateFormat = config.getDateFormat();
info.setDatetime(now);
info.setDtFormat(dateFormat);
File qrcodeFile = context.getQrcodeFile();
// 从 config 读取二维码文件
File qrcodeFile = config.getQrcodeFile();
if (qrcodeFile != null && qrcodeFile.exists()) {
info.setQrcodeFile(qrcodeFile);
}
Integer offsetLeft = context.getOffsetLeft();
if (offsetLeft != null) {
info.setOffsetLeft(offsetLeft);
// 根据旋转状态自己处理 offsetLeft
if (context.isNeedRotation()) {
info.setOffsetLeft(OFFSET_LEFT_FOR_PORTRAIT);
}
context.setWatermarkInfo(info);
return info;
}
}

View File

@@ -17,6 +17,7 @@ import com.ycwl.basic.image.pipeline.stages.ImageOrientationStage;
import com.ycwl.basic.image.pipeline.stages.PuzzleBorderStage;
import com.ycwl.basic.image.pipeline.stages.RestoreOrientationStage;
import com.ycwl.basic.image.pipeline.stages.UploadStage;
import com.ycwl.basic.image.pipeline.stages.WatermarkConfig;
import com.ycwl.basic.image.pipeline.stages.WatermarkStage;
import com.ycwl.basic.image.watermark.enums.ImageWatermarkOperatorEnum;
import com.ycwl.basic.integration.common.manager.DeviceConfigManager;
@@ -742,13 +743,15 @@ public class PrinterServiceImpl implements PrinterService {
/**
* 创建普通照片处理管线
*
* @param watermarkConfig 水印配置
*/
private Pipeline<PhotoProcessContext> createNormalPhotoPipeline() {
private Pipeline<PhotoProcessContext> createNormalPhotoPipeline(WatermarkConfig watermarkConfig) {
return new PipelineBuilder<PhotoProcessContext>("NormalPhotoPipeline")
.addStage(new DownloadStage())
.addStage(new ImageOrientationStage())
.addStage(new ConditionalRotateStage())
.addStage(new WatermarkStage())
.addStage(new WatermarkStage(watermarkConfig))
.addStage(new RestoreOrientationStage())
.addStage(new UploadStage())
.addStage(new CleanupStage())
@@ -778,7 +781,6 @@ public class PrinterServiceImpl implements PrinterService {
PrinterOrderItem orderItem = PrinterOrderItem.fromMemberPrintResp(item);
PhotoProcessContext context = PhotoProcessContext.fromPrinterOrderItem(orderItem, scenicId);
context.setQrcodeFile(qrCodeFile);
try {
// 设置景区配置管理器到context
@@ -800,16 +802,20 @@ public class PrinterServiceImpl implements PrinterService {
context.setSource(ImageSource.UNKNOWN);
}
Pipeline<PhotoProcessContext> pipeline;
if (context.getImageType() == ImageType.NORMAL_PHOTO) {
prepareNormalPhotoContext(context);
} else if (context.getImageType() == ImageType.PUZZLE) {
// 准备水印配置
WatermarkConfig watermarkConfig = prepareWatermarkConfig(context, qrCodeFile);
// 准备存储适配器
prepareStorageAdapter(context);
// 创建普通照片管线
pipeline = createNormalPhotoPipeline(watermarkConfig);
} else {
// 拼图
prepareStorageAdapter(context);
pipeline = createPuzzlePipeline();
}
Pipeline<PhotoProcessContext> pipeline = context.getImageType() == ImageType.PUZZLE
? createPuzzlePipeline()
: createNormalPhotoPipeline();
boolean success = pipeline.execute(context);
if (success && context.getResultUrl() != null) {
@@ -832,29 +838,35 @@ public class PrinterServiceImpl implements PrinterService {
}
/**
* 准备普通照片的Context配置
* 从context中的scenicConfigManager获取配置
* 准备普通照片的水印配置
* 从 scenicConfigManager 获取配置并构建 WatermarkConfig
*
* @param context 照片处理上下文
* @param qrCodeFile 二维码文件
* @return WatermarkConfig
*/
private void prepareNormalPhotoContext(PhotoProcessContext context) {
private WatermarkConfig prepareWatermarkConfig(PhotoProcessContext context, File qrCodeFile) {
ScenicConfigManager scenicConfig = context.getScenicConfigManager();
if (scenicConfig == null) {
log.warn("scenicConfigManager未设置,跳过配置准备");
return;
log.warn("scenicConfigManager未设置,返回空水印配置");
return WatermarkConfig.builder().build();
}
ImageWatermarkOperatorEnum watermarkType = null;
String printWatermarkType = scenicConfig.getString("print_watermark_type");
if (StringUtils.isNotBlank(printWatermarkType)) {
ImageWatermarkOperatorEnum watermarkType = ImageWatermarkOperatorEnum.getByCode(printWatermarkType);
context.setWatermarkType(watermarkType);
watermarkType = ImageWatermarkOperatorEnum.getByCode(printWatermarkType);
}
String scenicText = scenicConfig.getString("print_watermark_scenic_text", "");
context.setScenicText(scenicText);
String dateFormat = scenicConfig.getString("print_watermark_dt_format", "yyyy.MM.dd");
context.setDateFormat(dateFormat);
prepareStorageAdapter(context);
return WatermarkConfig.builder()
.watermarkType(watermarkType)
.scenicText(scenicText)
.dateFormat(dateFormat)
.qrcodeFile(qrCodeFile)
.build();
}
/**

View File

@@ -3,6 +3,7 @@ package com.ycwl.basic.image.pipeline.stages;
import com.ycwl.basic.image.pipeline.core.PhotoProcessContext;
import com.ycwl.basic.image.pipeline.core.StageResult;
import com.ycwl.basic.image.pipeline.enums.ImageType;
import com.ycwl.basic.image.watermark.enums.ImageWatermarkOperatorEnum;
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;
@@ -14,14 +15,18 @@ class WatermarkStageTest {
@Test
void testGetName() {
WatermarkStage stage = new WatermarkStage();
WatermarkConfig config = WatermarkConfig.builder().build();
WatermarkStage stage = new WatermarkStage(config);
assertEquals("WatermarkStage", stage.getName());
}
@Test
void testShouldExecute_NormalPhoto() {
WatermarkStage stage = new WatermarkStage();
WatermarkConfig config = WatermarkConfig.builder()
.watermarkType(ImageWatermarkOperatorEnum.PRINTER_DEFAULT)
.build();
WatermarkStage stage = new WatermarkStage(config);
PhotoProcessContext context = PhotoProcessContext.builder()
.originalUrl("https://example.com/test.jpg")
@@ -34,7 +39,10 @@ class WatermarkStageTest {
@Test
void testShouldExecute_Puzzle_ShouldSkip() {
WatermarkStage stage = new WatermarkStage();
WatermarkConfig config = WatermarkConfig.builder()
.watermarkType(ImageWatermarkOperatorEnum.PRINTER_DEFAULT)
.build();
WatermarkStage stage = new WatermarkStage(config);
PhotoProcessContext context = PhotoProcessContext.builder()
.originalUrl("https://example.com/puzzle.png")
@@ -47,7 +55,10 @@ class WatermarkStageTest {
@Test
void testShouldExecute_MobileUpload_ShouldSkip() {
WatermarkStage stage = new WatermarkStage();
WatermarkConfig config = WatermarkConfig.builder()
.watermarkType(ImageWatermarkOperatorEnum.PRINTER_DEFAULT)
.build();
WatermarkStage stage = new WatermarkStage(config);
PhotoProcessContext context = PhotoProcessContext.builder()
.originalUrl("https://example.com/mobile.jpg")
@@ -60,7 +71,9 @@ class WatermarkStageTest {
@Test
void testExecute_NoWatermarkType_ShouldSkip() {
WatermarkStage stage = new WatermarkStage();
// 创建没有 watermarkType 的配置
WatermarkConfig config = WatermarkConfig.builder().build();
WatermarkStage stage = new WatermarkStage(config);
PhotoProcessContext context = PhotoProcessContext.builder()
.originalUrl("https://example.com/test.jpg")