From 5a34fb51a8155c010212514345fe7fe0cd2981c6 Mon Sep 17 00:00:00 2001 From: Halo Dev Bot <87291978+halo-dev-bot@users.noreply.github.com> Date: Fri, 22 Sep 2023 17:56:23 +0800 Subject: [PATCH] [release-2.9] fix: not using the default template when the custom template does not exist (#4648) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This is an automated cherry-pick of #4618 /assign guqing ```release-note 修复文章自定义模板不存在时没有使用默认模板的问题 ``` --- .../app/theme/router/ViewNameResolver.java | 35 ++++++----- .../theme/router/ViewNameResolverTest.java | 58 +++++++++++++------ 2 files changed, 58 insertions(+), 35 deletions(-) diff --git a/application/src/main/java/run/halo/app/theme/router/ViewNameResolver.java b/application/src/main/java/run/halo/app/theme/router/ViewNameResolver.java index 38e7bdf15b..de53492ca3 100644 --- a/application/src/main/java/run/halo/app/theme/router/ViewNameResolver.java +++ b/application/src/main/java/run/halo/app/theme/router/ViewNameResolver.java @@ -1,14 +1,14 @@ package run.halo.app.theme.router; -import java.util.Locale; +import java.nio.file.Files; import lombok.AllArgsConstructor; import org.apache.commons.lang3.StringUtils; import org.springframework.boot.autoconfigure.thymeleaf.ThymeleafProperties; -import org.springframework.context.i18n.LocaleContextHolder; import org.springframework.stereotype.Component; +import org.springframework.util.Assert; import org.springframework.web.reactive.function.server.ServerRequest; import reactor.core.publisher.Mono; -import run.halo.app.theme.HaloViewResolver; +import run.halo.app.theme.ThemeResolver; /** * The {@link ViewNameResolver} is used to resolve view name. @@ -19,7 +19,8 @@ @Component @AllArgsConstructor public class ViewNameResolver { - private final HaloViewResolver haloViewResolver; + private static final String TEMPLATES = "templates"; + private final ThemeResolver themeResolver; private final ThymeleafProperties thymeleafProperties; /** @@ -29,20 +30,22 @@ public class ViewNameResolver { public Mono resolveViewNameOrDefault(ServerRequest request, String name, String defaultName) { if (StringUtils.isBlank(name)) { - return Mono.just(defaultName); + return Mono.justOrEmpty(defaultName); } - final String nameToUse = processName(name); - Locale locale = LocaleContextHolder.getLocale(request.exchange().getLocaleContext()); - return haloViewResolver.resolveViewName(nameToUse, locale) - .map(view -> nameToUse) - .switchIfEmpty(Mono.just(defaultName)); + return themeResolver.getTheme(request.exchange()) + .mapNotNull(themeContext -> { + String templateResourceName = computeResourceName(name); + var resourcePath = themeContext.getPath() + .resolve(TEMPLATES) + .resolve(templateResourceName); + return Files.exists(resourcePath) ? name : defaultName; + }) + .switchIfEmpty(Mono.justOrEmpty(defaultName)); } - String processName(String name) { - String nameToLookup = name; - if (StringUtils.endsWith(name, thymeleafProperties.getSuffix())) { - nameToLookup = StringUtils.substringBeforeLast(name, thymeleafProperties.getSuffix()); - } - return nameToLookup; + String computeResourceName(String name) { + Assert.notNull(name, "Name must not be null"); + return StringUtils.endsWith(name, thymeleafProperties.getSuffix()) + ? name : name + thymeleafProperties.getSuffix(); } } diff --git a/application/src/test/java/run/halo/app/theme/router/ViewNameResolverTest.java b/application/src/test/java/run/halo/app/theme/router/ViewNameResolverTest.java index f069e7dfbe..8ec9e863c5 100644 --- a/application/src/test/java/run/halo/app/theme/router/ViewNameResolverTest.java +++ b/application/src/test/java/run/halo/app/theme/router/ViewNameResolverTest.java @@ -1,15 +1,19 @@ package run.halo.app.theme.router; import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.assertThatThrownBy; import static org.mockito.ArgumentMatchers.any; -import static org.mockito.ArgumentMatchers.eq; import static org.mockito.Mockito.when; +import java.io.File; +import java.io.IOException; import java.net.URI; import java.net.URISyntaxException; +import java.nio.file.Files; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.ExtendWith; +import org.junit.jupiter.api.io.TempDir; import org.mockito.InjectMocks; import org.mockito.Mock; import org.mockito.Mockito; @@ -17,11 +21,11 @@ import org.springframework.http.HttpMethod; import org.springframework.mock.web.reactive.function.server.MockServerRequest; import org.springframework.test.context.junit.jupiter.SpringExtension; -import org.springframework.web.reactive.result.view.View; import org.springframework.web.server.ServerWebExchange; import reactor.core.publisher.Mono; import reactor.test.StepVerifier; -import run.halo.app.theme.HaloViewResolver; +import run.halo.app.theme.ThemeContext; +import run.halo.app.theme.ThemeResolver; /** * Tests for {@link ViewNameResolver}. @@ -33,7 +37,7 @@ class ViewNameResolverTest { @Mock - private HaloViewResolver haloViewResolver; + private ThemeResolver themeResolver; @Mock private ThymeleafProperties thymeleafProperties; @@ -41,17 +45,27 @@ class ViewNameResolverTest { @InjectMocks private ViewNameResolver viewNameResolver; + @TempDir + private File themePath; + @BeforeEach - void setUp() { + void setUp() throws IOException { when(thymeleafProperties.getSuffix()).thenReturn(ThymeleafProperties.DEFAULT_SUFFIX); - when(haloViewResolver.resolveViewName(eq("post_news"), any())) - .thenReturn(Mono.just(Mockito.mock(View.class))); - when(haloViewResolver.resolveViewName(eq("post_docs"), any())) - .thenReturn(Mono.just(new EmptyView())); + var templatesPath = themePath.toPath().resolve("templates"); + if (!Files.exists(templatesPath)) { + Files.createDirectory(templatesPath); + } + Files.createFile(templatesPath.resolve("post_news.html")); + Files.createFile(templatesPath.resolve("post_docs.html")); - when(haloViewResolver.resolveViewName(eq("post_nothing"), any())) - .thenReturn(Mono.empty()); + when(themeResolver.getTheme(any())) + .thenReturn(Mono.fromSupplier(() -> ThemeContext.builder() + .name("fake-theme") + .path(themePath.toPath()) + .active(true) + .build()) + ); } @Test @@ -71,7 +85,7 @@ void resolveViewNameOrDefault() throws URISyntaxException { String viewName = "post_docs" + thymeleafProperties.getSuffix(); viewNameResolver.resolveViewNameOrDefault(request, viewName, "post") .as(StepVerifier::create) - .expectNext("post_docs") + .expectNext(viewName) .verifyComplete(); viewNameResolver.resolveViewNameOrDefault(request, "post_nothing", "post") @@ -82,11 +96,17 @@ void resolveViewNameOrDefault() throws URISyntaxException { @Test void processName() { - assertThat(viewNameResolver.processName("post_news")).isEqualTo("post_news"); - assertThat(viewNameResolver.processName("post_news" + thymeleafProperties.getSuffix())) - .isEqualTo("post_news"); - assertThat(viewNameResolver.processName("post_news.test")) - .isEqualTo("post_news.test"); - assertThat(viewNameResolver.processName(null)).isNull(); + var suffix = thymeleafProperties.getSuffix(); + assertThat(viewNameResolver.computeResourceName("post_news")) + .isEqualTo("post_news" + suffix); + assertThat( + viewNameResolver.computeResourceName("post_news" + suffix)) + .isEqualTo("post_news" + suffix); + assertThat(viewNameResolver.computeResourceName("post_news.test")) + .isEqualTo("post_news.test" + suffix); + + assertThatThrownBy(() -> viewNameResolver.computeResourceName(null)) + .isInstanceOf(IllegalArgumentException.class) + .hasMessage("Name must not be null"); } -} \ No newline at end of file +}