From 69ed005e0ed7772a01ce0da7e70fcfdc62455fd7 Mon Sep 17 00:00:00 2001 From: guqing Date: Mon, 12 Aug 2024 16:35:50 +0800 Subject: [PATCH] feat: ability to configure and use image thumbnails --- build.gradle | 3 +- .../s3os/AttachmentThumbnailReconciler.java | 74 ++++++++++ .../java/run/halo/s3os/S3LinkServiceImpl.java | 4 +- .../run/halo/s3os/S3OsAttachmentHandler.java | 49 ++----- src/main/java/run/halo/s3os/S3OsPlugin.java | 1 + .../java/run/halo/s3os/S3OsProperties.java | 42 +++++- .../run/halo/s3os/S3ThumbnailProvider.java | 128 ++++++++++++++++++ .../resources/extensions/ext-definitions.yaml | 9 ++ .../extensions/policy-template-s3os.yaml | 43 +++++- .../AttachmentThumbnailReconcilerTest.java | 25 ++++ 10 files changed, 327 insertions(+), 51 deletions(-) create mode 100644 src/main/java/run/halo/s3os/AttachmentThumbnailReconciler.java create mode 100644 src/main/java/run/halo/s3os/S3ThumbnailProvider.java create mode 100644 src/main/resources/extensions/ext-definitions.yaml create mode 100644 src/test/java/run/halo/s3os/AttachmentThumbnailReconcilerTest.java diff --git a/build.gradle b/build.gradle index 92bcde0..097ca60 100644 --- a/build.gradle +++ b/build.gradle @@ -13,6 +13,7 @@ java { } repositories { + mavenLocal() mavenCentral() maven { url 'https://s01.oss.sonatype.org/content/repositories/releases' } maven { url 'https://s01.oss.sonatype.org/content/repositories/snapshots/' } @@ -20,7 +21,7 @@ repositories { } dependencies { - implementation platform('run.halo.tools.platform:plugin:2.17.0-SNAPSHOT') + implementation platform('run.halo.tools.platform:plugin:2.19.0-SNAPSHOT') compileOnly 'run.halo.app:api' implementation platform('software.amazon.awssdk:bom:2.19.8') diff --git a/src/main/java/run/halo/s3os/AttachmentThumbnailReconciler.java b/src/main/java/run/halo/s3os/AttachmentThumbnailReconciler.java new file mode 100644 index 0000000..d662174 --- /dev/null +++ b/src/main/java/run/halo/s3os/AttachmentThumbnailReconciler.java @@ -0,0 +1,74 @@ +package run.halo.s3os; + +import static run.halo.app.infra.FileCategoryMatcher.IMAGE; + +import java.time.Duration; +import java.time.Instant; +import lombok.RequiredArgsConstructor; +import org.apache.commons.lang3.StringUtils; +import org.springframework.stereotype.Component; +import org.springframework.util.Assert; +import run.halo.app.core.extension.attachment.Attachment; +import run.halo.app.extension.ExtensionClient; +import run.halo.app.extension.MetadataUtil; +import run.halo.app.extension.controller.Controller; +import run.halo.app.extension.controller.ControllerBuilder; +import run.halo.app.extension.controller.Reconciler; + +/** + *

This {@link Reconciler} used to check thumbnail status are generated + * if not, update the attachment to trigger thumbnail generation by halo

+ */ +@Component +@RequiredArgsConstructor +public class AttachmentThumbnailReconciler implements Reconciler { + static final String REQUEST_GEN_THUMBNAIL = "s3os.halo.run/request-gen-thumbnail"; + private final ExtensionClient client; + + @Override + public Result reconcile(Request request) { + client.fetch(Attachment.class, request.name()) + .ifPresent(attachment -> { + var annotations = MetadataUtil.nullSafeAnnotations(attachment); + var requestTime = annotations.get(REQUEST_GEN_THUMBNAIL); + if (isMadeWithIn1Day(requestTime)) { + // skip if request is made within 1 day + return; + } + + if (!isImage(attachment)) { + // skip non-image attachments + return; + } + + var status = attachment.getStatus(); + if (status == null || status.getThumbnails() == null) { + // update to trigger attachment thumbnail generation + annotations.put(REQUEST_GEN_THUMBNAIL, Instant.now().toString()); + client.update(attachment); + } + }); + return Result.doNotRetry(); + } + + static boolean isMadeWithIn1Day(String requestTimeStr) { + if (StringUtils.isBlank(requestTimeStr)) { + return false; + } + var requestTime = Instant.parse(requestTimeStr); + return Duration.between(requestTime, Instant.now()).minusDays(1).isNegative(); + } + + public static boolean isImage(Attachment attachment) { + Assert.notNull(attachment, "Attachment must not be null"); + var mediaType = attachment.getSpec().getMediaType(); + return mediaType != null && IMAGE.match(mediaType); + } + + @Override + public Controller setupWith(ControllerBuilder builder) { + return builder + .extension(new Attachment()) + .build(); + } +} diff --git a/src/main/java/run/halo/s3os/S3LinkServiceImpl.java b/src/main/java/run/halo/s3os/S3LinkServiceImpl.java index ba6b6d1..8e81459 100644 --- a/src/main/java/run/halo/s3os/S3LinkServiceImpl.java +++ b/src/main/java/run/halo/s3os/S3LinkServiceImpl.java @@ -67,7 +67,7 @@ public Mono listObjects(String policyName, String continuationToke return client.fetch(ConfigMap.class, configMapName); }) .flatMap((configMap) -> { - var properties = handler.getProperties(configMap); + var properties = S3OsProperties.convertFrom(configMap); var finalLocation = FilePathUtils.getFilePathByPlaceholder(properties.getLocation()); return Mono.using(() -> handler.buildS3Client(properties), // 执行 listObjects @@ -231,7 +231,7 @@ public Mono addAttachmentRecord(String policyName, return client.fetch(ConfigMap.class, configMapName); }) .flatMap(configMap -> { - var properties = handler.getProperties(configMap); + var properties = S3OsProperties.convertFrom(configMap); return Mono.using(() -> handler.buildS3Client(properties), (s3Client) -> Mono.fromCallable( () -> s3Client.headObject( diff --git a/src/main/java/run/halo/s3os/S3OsAttachmentHandler.java b/src/main/java/run/halo/s3os/S3OsAttachmentHandler.java index c2db9c3..ca57fb1 100644 --- a/src/main/java/run/halo/s3os/S3OsAttachmentHandler.java +++ b/src/main/java/run/halo/s3os/S3OsAttachmentHandler.java @@ -3,7 +3,6 @@ import java.net.URI; import java.net.URISyntaxException; import java.nio.ByteBuffer; -import java.nio.charset.StandardCharsets; import java.nio.file.FileAlreadyExistsException; import java.time.Duration; import java.util.HashMap; @@ -23,7 +22,6 @@ import org.springframework.lang.Nullable; import org.springframework.web.server.ServerErrorException; import org.springframework.web.server.ServerWebInputException; -import org.springframework.web.util.UriUtils; import reactor.core.Exceptions; import reactor.core.publisher.Flux; import reactor.core.publisher.Mono; @@ -38,7 +36,6 @@ import run.halo.app.extension.ConfigMap; import run.halo.app.extension.Metadata; import run.halo.app.extension.MetadataUtil; -import run.halo.app.infra.utils.JsonUtils; import software.amazon.awssdk.auth.credentials.AwsBasicCredentials; import software.amazon.awssdk.awscore.presigner.SdkPresigner; import software.amazon.awssdk.core.SdkResponse; @@ -47,16 +44,7 @@ import software.amazon.awssdk.regions.Region; import software.amazon.awssdk.services.s3.S3Client; import software.amazon.awssdk.services.s3.S3Configuration; -import software.amazon.awssdk.services.s3.model.CompleteMultipartUploadRequest; -import software.amazon.awssdk.services.s3.model.CompletedMultipartUpload; -import software.amazon.awssdk.services.s3.model.CompletedPart; -import software.amazon.awssdk.services.s3.model.CreateMultipartUploadRequest; -import software.amazon.awssdk.services.s3.model.DeleteObjectRequest; -import software.amazon.awssdk.services.s3.model.GetObjectRequest; -import software.amazon.awssdk.services.s3.model.HeadObjectRequest; -import software.amazon.awssdk.services.s3.model.HeadObjectResponse; -import software.amazon.awssdk.services.s3.model.NoSuchKeyException; -import software.amazon.awssdk.services.s3.model.UploadPartRequest; +import software.amazon.awssdk.services.s3.model.*; import software.amazon.awssdk.services.s3.presigner.S3Presigner; import software.amazon.awssdk.services.s3.presigner.model.GetObjectPresignRequest; import software.amazon.awssdk.utils.SdkAutoCloseable; @@ -79,7 +67,7 @@ public class S3OsAttachmentHandler implements AttachmentHandler { public Mono upload(UploadContext uploadContext) { return Mono.just(uploadContext).filter(context -> this.shouldHandle(context.policy())) .flatMap(context -> { - final var properties = getProperties(context.configMap()); + final var properties = S3OsProperties.convertFrom(context.configMap()); return upload(context, properties) .subscribeOn(Schedulers.boundedElastic()) .map(objectDetail -> this.buildAttachment(properties, objectDetail)) @@ -102,7 +90,7 @@ public Mono delete(DeleteContext deleteContext) { log.info("Skip deleting object {} from S3.", objectKey); return Mono.just(context); } - var properties = getProperties(deleteContext.configMap()); + var properties = S3OsProperties.convertFrom(deleteContext.configMap()); return Mono.using(() -> buildS3Client(properties), client -> Mono.fromCallable( () -> client.deleteObject(DeleteObjectRequest.builder() @@ -123,7 +111,7 @@ public Mono delete(DeleteContext deleteContext) { @Override public Mono getSharedURL(Attachment attachment, Policy policy, ConfigMap configMap, - Duration ttl) { + Duration ttl) { if (!this.shouldHandle(policy)) { return Mono.empty(); } @@ -132,7 +120,7 @@ public Mono getSharedURL(Attachment attachment, Policy policy, ConfigMap co return Mono.error(new IllegalArgumentException( "Cannot obtain object key from attachment " + attachment.getMetadata().getName())); } - var properties = getProperties(configMap); + var properties = S3OsProperties.convertFrom(configMap); return Mono.using(() -> buildS3Presigner(properties), s3Presigner -> { @@ -168,8 +156,8 @@ public Mono getPermalink(Attachment attachment, Policy policy, ConfigMap co // fallback to default handler for backward compatibility return Mono.empty(); } - var properties = getProperties(configMap); - var objectURL = getObjectURL(properties, objectKey); + var properties = S3OsProperties.convertFrom(configMap); + var objectURL = properties.toObjectURL(objectKey); var urlSuffix = getUrlSuffixAnnotation(attachment); if (StringUtils.isNotBlank(urlSuffix)) { objectURL += urlSuffix; @@ -195,13 +183,8 @@ private String getUrlSuffixAnnotation(Attachment attachment) { return annotations.get(URL_SUFFIX_ANNO_KEY); } - S3OsProperties getProperties(ConfigMap configMap) { - var settingJson = configMap.getData().getOrDefault("default", "{}"); - return JsonUtils.jsonToObject(settingJson, S3OsProperties.class); - } - Attachment buildAttachment(S3OsProperties properties, ObjectDetail objectDetail) { - String externalLink = getObjectURL(properties, objectDetail.uploadState.objectKey); + String externalLink = properties.toObjectURL(objectDetail.uploadState.objectKey); var urlSuffix = UrlUtils.findUrlSuffix(properties.getUrlSuffixes(), objectDetail.uploadState.fileName); @@ -229,22 +212,6 @@ Attachment buildAttachment(S3OsProperties properties, ObjectDetail objectDetail) return attachment; } - String getObjectURL(S3OsProperties properties, String objectKey) { - String objectURL; - if (StringUtils.isBlank(properties.getDomain())) { - String host; - if (properties.getEnablePathStyleAccess()) { - host = properties.getEndpoint() + "/" + properties.getBucket(); - } else { - host = properties.getBucket() + "." + properties.getEndpoint(); - } - objectURL = properties.getProtocol() + "://" + host + "/" + objectKey; - } else { - objectURL = properties.getProtocol() + "://" + properties.getDomain() + "/" + objectKey; - } - return UriUtils.encodePath(objectURL, StandardCharsets.UTF_8); - } - S3Client buildS3Client(S3OsProperties properties) { return S3Client.builder() .region(Region.of(properties.getRegion())) diff --git a/src/main/java/run/halo/s3os/S3OsPlugin.java b/src/main/java/run/halo/s3os/S3OsPlugin.java index 7851027..5c3cdbd 100644 --- a/src/main/java/run/halo/s3os/S3OsPlugin.java +++ b/src/main/java/run/halo/s3os/S3OsPlugin.java @@ -10,6 +10,7 @@ */ @Component public class S3OsPlugin extends BasePlugin { + public static final String POLICY_SETTING_NAME = "s3os-policy-template-setting"; public S3OsPlugin(PluginContext pluginContext) { super(pluginContext); diff --git a/src/main/java/run/halo/s3os/S3OsProperties.java b/src/main/java/run/halo/s3os/S3OsProperties.java index 634eea2..b87c7da 100644 --- a/src/main/java/run/halo/s3os/S3OsProperties.java +++ b/src/main/java/run/halo/s3os/S3OsProperties.java @@ -1,16 +1,17 @@ package run.halo.s3os; +import java.nio.charset.StandardCharsets; import java.util.List; import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor; -import lombok.RequiredArgsConstructor; import org.springframework.util.StringUtils; - -import java.time.LocalDate; +import org.springframework.web.util.UriUtils; +import run.halo.app.extension.ConfigMap; +import run.halo.app.infra.utils.JsonUtils; @Data -class S3OsProperties { +public class S3OsProperties { private String bucket; @@ -49,6 +50,14 @@ class S3OsProperties { private List urlSuffixes; + private ThumbnailParam thumbnailParam; + + public record ThumbnailParam(String type, String pattern) { + public boolean hasPattern() { + return StringUtils.hasText(type) && StringUtils.hasText(pattern); + } + } + @Data @AllArgsConstructor @NoArgsConstructor @@ -103,14 +112,14 @@ public void setRandomStringLength(String randomStringLength) { // if you use In if (length >= 4 && length <= 16) { this.randomStringLength = length; } + } catch (NumberFormatException ignored) { } - catch (NumberFormatException ignored) { } } public void setRegion(String region) { if (!StringUtils.hasText(region)) { this.region = "Auto"; - }else { + } else { this.region = region; } } @@ -118,4 +127,25 @@ public void setRegion(String region) { public void setEndpoint(String endpoint) { this.endpoint = UrlUtils.removeHttpPrefix(endpoint); } + + public String toObjectURL(String objectKey) { + String objectURL; + if (!StringUtils.hasText(this.getDomain())) { + String host; + if (this.getEnablePathStyleAccess()) { + host = this.getEndpoint() + "/" + this.getBucket(); + } else { + host = this.getBucket() + "." + this.getEndpoint(); + } + objectURL = this.getProtocol() + "://" + host + "/" + objectKey; + } else { + objectURL = this.getProtocol() + "://" + this.getDomain() + "/" + objectKey; + } + return UriUtils.encodePath(objectURL, StandardCharsets.UTF_8); + } + + public static S3OsProperties convertFrom(ConfigMap configMap) { + var settingJson = configMap.getData().getOrDefault("default", "{}"); + return JsonUtils.jsonToObject(settingJson, S3OsProperties.class); + } } diff --git a/src/main/java/run/halo/s3os/S3ThumbnailProvider.java b/src/main/java/run/halo/s3os/S3ThumbnailProvider.java new file mode 100644 index 0000000..0e6ac8c --- /dev/null +++ b/src/main/java/run/halo/s3os/S3ThumbnailProvider.java @@ -0,0 +1,128 @@ +package run.halo.s3os; + +import static run.halo.s3os.S3OsPlugin.POLICY_SETTING_NAME; + +import java.net.URI; +import java.net.URL; +import java.util.List; +import java.util.Map; +import com.google.common.cache.Cache; +import com.google.common.cache.CacheBuilder; +import lombok.Builder; +import lombok.RequiredArgsConstructor; +import org.apache.commons.lang3.StringUtils; +import org.springframework.data.domain.Sort; +import org.springframework.stereotype.Component; +import org.springframework.util.Assert; +import org.springframework.web.util.UriComponentsBuilder; +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; +import run.halo.app.core.attachment.ThumbnailProvider; +import run.halo.app.core.attachment.ThumbnailSize; +import run.halo.app.core.extension.attachment.Policy; +import run.halo.app.core.extension.attachment.PolicyTemplate; +import run.halo.app.extension.ConfigMap; +import run.halo.app.extension.ListOptions; +import run.halo.app.extension.ReactiveExtensionClient; + +@Component +@RequiredArgsConstructor +public class S3ThumbnailProvider implements ThumbnailProvider { + static final String WIDTH_PLACEHOLDER = "{width}"; + private final Cache s3PropsCache = CacheBuilder.newBuilder() + .maximumSize(50) + .build(); + + private final ReactiveExtensionClient client; + + @Override + public Mono generate(ThumbnailContext thumbnailContext) { + var url = thumbnailContext.getImageUrl().toString(); + var size = thumbnailContext.getSize(); + return getCacheValue(url) + .mapNotNull(cacheValue -> placedPattern(cacheValue.pattern(), size)) + .map(param -> { + if (param.startsWith("?")) { + UriComponentsBuilder.fromHttpUrl(url) + .queryParam(param.substring(1)); + } + return url + param; + }) + .map(URI::create); + } + + private static String placedPattern(String pattern, ThumbnailSize size) { + return StringUtils.replace(pattern, WIDTH_PLACEHOLDER, String.valueOf(size.getWidth())); + } + + @Override + public Mono delete(URL url) { + // do nothing for s3 + return Mono.empty(); + } + + @Override + public Mono supports(ThumbnailContext thumbnailContext) { + var url = thumbnailContext.getImageUrl().toString(); + return getCacheValue(url).hasElement(); + } + + private Mono getCacheValue(String imageUrl) { + return Flux.fromIterable(s3PropsCache.asMap().entrySet()) + .filter(entry -> imageUrl.startsWith(entry.getKey())) + .next() + .map(Map.Entry::getValue) + .switchIfEmpty(Mono.defer(() -> listAllS3ObjectDomain() + .filter(entry -> imageUrl.startsWith(entry.getKey())) + .map(Map.Entry::getValue) + .next() + )); + } + + @Builder + record S3PropsCacheValue(String pattern, String configMapName) { + } + + private Flux> listAllS3ObjectDomain() { + return listS3PolicyTemplateNames() + .collectList() + .flatMapMany(this::listAllS3Policy) + .flatMap(s3Policy -> { + var s3ConfigMapName = s3Policy.getSpec().getConfigMapName(); + return fetchS3PropsByConfigMapName(s3ConfigMapName) + .mapNotNull(properties -> { + var thumbnailParam = properties.getThumbnailParam(); + if (thumbnailParam == null || !thumbnailParam.hasPattern()) { + return null; + } + var objectDomain = properties.toObjectURL(""); + var cacheValue = S3PropsCacheValue.builder() + .pattern(thumbnailParam.pattern()) + .configMapName(s3ConfigMapName) + .build(); + return Map.entry(objectDomain, cacheValue); + }); + }) + .doOnNext(cache -> s3PropsCache.put(cache.getKey(), cache.getValue())); + } + + private Flux listAllS3Policy(List s3PolicyTemplateNames) { + Assert.notNull(s3PolicyTemplateNames, "The s3PolicyTemplateNames must not be null."); + return client.listAll(Policy.class, new ListOptions(), Sort.unsorted()) + .filter(policy -> { + var templateName = policy.getSpec().getTemplateName(); + return s3PolicyTemplateNames.contains(templateName); + }); + } + + private Mono fetchS3PropsByConfigMapName(String name) { + return client.fetch(ConfigMap.class, name) + .map(S3OsProperties::convertFrom); + } + + private Flux listS3PolicyTemplateNames() { + return client.listAll(PolicyTemplate.class, new ListOptions(), Sort.unsorted()) + .filter(policyTemplate -> POLICY_SETTING_NAME.equals(policyTemplate.getSpec().getSettingName())) + .map(template -> template.getMetadata().getName()); + } +} diff --git a/src/main/resources/extensions/ext-definitions.yaml b/src/main/resources/extensions/ext-definitions.yaml new file mode 100644 index 0000000..fbc1d0c --- /dev/null +++ b/src/main/resources/extensions/ext-definitions.yaml @@ -0,0 +1,9 @@ +apiVersion: plugin.halo.run/v1alpha1 +kind: ExtensionDefinition +metadata: + name: s3os-thumbnail-provider +spec: + className: run.halo.s3os.S3ThumbnailProvider + extensionPointName: thumbnail-provider + displayName: "S3 协议 OSS 缩略图生成" + description: "为上传到支持 S3 协议的 OSS 的图片生成缩略图" diff --git a/src/main/resources/extensions/policy-template-s3os.yaml b/src/main/resources/extensions/policy-template-s3os.yaml index 80c7c01..f44e4c5 100644 --- a/src/main/resources/extensions/policy-template-s3os.yaml +++ b/src/main/resources/extensions/policy-template-s3os.yaml @@ -144,4 +144,45 @@ spec: name: urlSuffix label: 网址后缀 placeholder: 例如:?imageMogr2/format/webp - validation: required \ No newline at end of file + validation: required + - $formkit: group + label: 缩略图参数 + name: thumbnailParam + children: + - $formkit: select + name: type + key: type + label: 类型 + options: + - label: 无 + value: "" + - label: 预设参数 + value: preset + - label: 自定义参数 + value: custom + - $formkit: select + if: "$value.type == preset" + name: pattern + key: type + label: 使用预设参数 + help: 请根据您的对象存储服务商选择对应的缩略图参数 + options: + - label: 腾讯云 COS + value: "?imageView2/0/w/{width}" + - label: 七牛云 KODO + value: "?imageView2/0/w/{width}" + - label: 阿里云 OSS + value: "?x-oss-process=image/resize,w_{width},m_lfit" + - label: 百度云 BOS + value: "?x-bce-process=image/resize,m_lfit,w_{width}" + - label: 青云 OSS + value: "?image&action=resize:w_{width},m_2" + - label: 京东云 + value: "?x-oss-process=img/sw/{width}" + - label: 又拍云 + value: "!/fw/{width}" + - $formkit: text + if: "$value.type == custom" + label: 自定义参数 + help: "{width} 为宽度占位符将被替换为所需缩略图宽度值,如: 400,参数需要以 ? 开头,间隔符除外" + name: pattern diff --git a/src/test/java/run/halo/s3os/AttachmentThumbnailReconcilerTest.java b/src/test/java/run/halo/s3os/AttachmentThumbnailReconcilerTest.java new file mode 100644 index 0000000..20d2ca3 --- /dev/null +++ b/src/test/java/run/halo/s3os/AttachmentThumbnailReconcilerTest.java @@ -0,0 +1,25 @@ +package run.halo.s3os; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.time.Duration; +import java.time.Instant; +import org.junit.jupiter.api.Test; + +class AttachmentThumbnailReconcilerTest { + + @Test + void isMadeWithIn1DayTest() { + var madeTime = Instant.now().plusSeconds(2).toString(); + var result = AttachmentThumbnailReconciler.isMadeWithIn1Day(madeTime); + assertThat(result).isTrue(); + + madeTime = Instant.now().plus(Duration.ofDays(2)).toString(); + result = AttachmentThumbnailReconciler.isMadeWithIn1Day(madeTime); + assertThat(result).isTrue(); + + madeTime = Instant.now().minus(Duration.ofHours(25)).toString(); + result = AttachmentThumbnailReconciler.isMadeWithIn1Day(madeTime); + assertThat(result).isFalse(); + } +}