diff --git a/.github/actions/prepare-gradle-build/action.yml b/.github/actions/prepare-gradle-build/action.yml index db0065a590fc..6532ce75276a 100644 --- a/.github/actions/prepare-gradle-build/action.yml +++ b/.github/actions/prepare-gradle-build/action.yml @@ -42,12 +42,12 @@ runs: ${{ inputs.java-toolchain == 'true' && '17' || '' }} - name: Set Up Gradle With Read/Write Cache if: ${{ inputs.cache-read-only == 'false' }} - uses: gradle/actions/setup-gradle@94baf225fe0a508e581a564467443d0e2379123b # v4.3.0 + uses: gradle/actions/setup-gradle@06832c7b30a0129d7fb559bcc6e43d26f6374244 # v4.3.1 with: cache-read-only: false develocity-access-key: ${{ inputs.develocity-access-key }} - name: Set Up Gradle - uses: gradle/actions/setup-gradle@94baf225fe0a508e581a564467443d0e2379123b # v4.3.0 + uses: gradle/actions/setup-gradle@06832c7b30a0129d7fb559bcc6e43d26f6374244 # v4.3.1 with: develocity-access-key: ${{ inputs.develocity-access-key }} develocity-token-expiry: 4 diff --git a/.github/workflows/build-and-deploy-snapshot.yml b/.github/workflows/build-and-deploy-snapshot.yml index e6fc93dbf240..de0b80b689c2 100644 --- a/.github/workflows/build-and-deploy-snapshot.yml +++ b/.github/workflows/build-and-deploy-snapshot.yml @@ -2,7 +2,7 @@ name: Build and Deploy Snapshot on: push: branches: - - '3.4.x' + - main concurrency: group: ${{ github.workflow }}-${{ github.ref }} jobs: @@ -27,7 +27,7 @@ jobs: - name: Deploy uses: spring-io/artifactory-deploy-action@dc1913008c0599f0c4b1fdafb6ff3c502b3565ea # v0.0.2 with: - build-name: ${{ vars.COMMERCIAL && format('spring-boot-commercial-{0}', github.ref_name) || format('spring-boot-{0}', github.ref_name) }} + build-name: ${{ vars.COMMERCIAL && format('spring-boot-commercial-{0}', '3.5.x') || format('spring-boot-{0}', '3.5.x') }} folder: 'deployment-repository' password: ${{ vars.COMMERCIAL && secrets.COMMERCIAL_ARTIFACTORY_PASSWORD || secrets.ARTIFACTORY_PASSWORD }} project: ${{ vars.COMMERCIAL && 'spring' }} diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 8d7b1c544844..1f32c5dab41a 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -2,7 +2,7 @@ name: CI on: push: branches: - - '3.4.x' + - main jobs: ci: name: '${{ matrix.os.name}} | Java ${{ matrix.java.version}}' diff --git a/.github/workflows/release-milestone.yml b/.github/workflows/release-milestone.yml index 6c5fa56f1d78..25cfe8dee6ad 100644 --- a/.github/workflows/release-milestone.yml +++ b/.github/workflows/release-milestone.yml @@ -2,8 +2,8 @@ name: Release Milestone on: push: tags: - - v3.4.0-M[0-9] - - v3.4.0-RC[0-9] + - v3.5.0-M[0-9] + - v3.5.0-RC[0-9] concurrency: group: ${{ github.workflow }}-${{ github.ref }} jobs: @@ -56,7 +56,7 @@ jobs: runs-on: ${{ vars.UBUNTU_SMALL || 'ubuntu-latest' }} steps: - name: Set up JFrog CLI - uses: jfrog/setup-jfrog-cli@dff217c085c17666e8849ebdbf29c8fe5e3995e6 # v4.5.2 + uses: jfrog/setup-jfrog-cli@f748a0599171a192a2668afee8d0497f7c1069df # v4.5.6 env: JF_ENV_SPRING: ${{ secrets.JF_ARTIFACTORY_SPRING }} - name: Promote build diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index 8dbf7582be81..d76c6a97d5c6 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -2,7 +2,7 @@ name: Release on: push: tags: - - v3.4.[0-9]+ + - v3.5.[0-9]+ concurrency: group: ${{ github.workflow }}-${{ github.ref }} jobs: @@ -117,7 +117,7 @@ jobs: - name: Publish to SDKMAN! uses: ./.github/actions/publish-to-sdkman with: - make-default: true + make-default: false sdkman-consumer-key: ${{ secrets.SDKMAN_CONSUMER_KEY }} sdkman-consumer-token: ${{ secrets.SDKMAN_CONSUMER_TOKEN }} spring-boot-version: ${{ needs.build-and-stage-release.outputs.version }} diff --git a/.github/workflows/run-system-tests.yml b/.github/workflows/run-system-tests.yml index afa746c2153e..26d8c2ddafdc 100644 --- a/.github/workflows/run-system-tests.yml +++ b/.github/workflows/run-system-tests.yml @@ -2,7 +2,7 @@ name: Run System Tests on: push: branches: - - '3.4.x' + - main jobs: run-system-tests: name: 'Java ${{ matrix.java.version}}' diff --git a/.github/workflows/trigger-docs-build.yml b/.github/workflows/trigger-docs-build.yml index 193cca000341..e583dd88589a 100644 --- a/.github/workflows/trigger-docs-build.yml +++ b/.github/workflows/trigger-docs-build.yml @@ -1,7 +1,7 @@ name: Trigger Docs Build on: push: - branches: '3.4.x' + branches: main paths: [ 'antora/*' ] workflow_dispatch: inputs: diff --git a/.github/workflows/verify.yml b/.github/workflows/verify.yml index ffe2221a82af..9c7c4c4d6f01 100644 --- a/.github/workflows/verify.yml +++ b/.github/workflows/verify.yml @@ -55,7 +55,7 @@ jobs: if: ${{ !vars.COMMERCIAL }} uses: Homebrew/actions/setup-homebrew@7657c9512f50e1c35b640971116425935bab3eea - name: Set Up Gradle - uses: gradle/actions/setup-gradle@94baf225fe0a508e581a564467443d0e2379123b # v4.3.0 + uses: gradle/actions/setup-gradle@06832c7b30a0129d7fb559bcc6e43d26f6374244 # v4.3.1 with: cache-read-only: false - name: Configure Gradle Properties diff --git a/.idea/scopes/java.xml b/.idea/scopes/java.xml index a4576baff901..98172e56e6a7 100644 --- a/.idea/scopes/java.xml +++ b/.idea/scopes/java.xml @@ -1,3 +1,3 @@ - + \ No newline at end of file diff --git a/README.adoc b/README.adoc index b1434a6387ef..72fccc99920b 100755 --- a/README.adoc +++ b/README.adoc @@ -1,4 +1,4 @@ -= Spring Boot image:https://github.com/spring-projects/spring-boot/actions/workflows/build-and-deploy-snapshot.yml/badge.svg?branch=3.4.x["Build Status", link="https://github.com/spring-projects/spring-boot/actions/workflows/build-and-deploy-snapshot.yml?query=branch%3Amain"] image:https://badges.gitter.im/Join Chat.svg["Chat",link="https://gitter.im/spring-projects/spring-boot?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge"] image:https://img.shields.io/badge/Revved%20up%20by-Develocity-06A0CE?logo=Gradle&labelColor=02303A["Revved up by Develocity", link="https://ge.spring.io/scans?&search.rootProjectNames=Spring%20Boot%20Build&search.rootProjectNames=spring-boot-build"] += Spring Boot image:https://github.com/spring-projects/spring-boot/actions/workflows/build-and-deploy-snapshot.yml/badge.svg?branch=main["Build Status", link="https://github.com/spring-projects/spring-boot/actions/workflows/build-and-deploy-snapshot.yml?query=branch%3Amain"] image:https://img.shields.io/badge/Revved%20up%20by-Develocity-06A0CE?logo=Gradle&labelColor=02303A["Revved up by Develocity", link="https://ge.spring.io/scans?&search.rootProjectNames=Spring%20Boot%20Build&search.rootProjectNames=spring-boot-build"] :docs: https://docs.spring.io/spring-boot :github: https://github.com/spring-projects/spring-boot @@ -57,7 +57,6 @@ Are you having trouble with Spring Boot? We want to help! If you are new to Spring, try one of the https://spring.io/guides[guides]. * If you are upgrading, read the {github}/wiki[release notes] for upgrade instructions and "new and noteworthy" features. * Ask a question -- we monitor https://stackoverflow.com[stackoverflow.com] for questions tagged with https://stackoverflow.com/tags/spring-boot[`spring-boot`]. - You can also chat with the community on https://gitter.im/spring-projects/spring-boot[Gitter]. * Report bugs with Spring Boot at {github}/issues[github.com/spring-projects/spring-boot/issues]. diff --git a/buildSrc/build.gradle b/buildSrc/build.gradle index d998c7306783..e0dfdcbf39ba 100644 --- a/buildSrc/build.gradle +++ b/buildSrc/build.gradle @@ -39,7 +39,7 @@ dependencies { implementation("io.spring.javaformat:spring-javaformat-gradle-plugin:${javaFormatVersion}") implementation("io.spring.nohttp:nohttp-gradle:0.0.11") implementation("org.apache.httpcomponents.client5:httpclient5:5.3.1") - implementation("org.apache.maven:maven-embedder:${mavenVersion}") + implementation("org.apache.maven:maven-artifact:${mavenVersion}") implementation("org.antora:gradle-antora-plugin:1.0.0") implementation("org.jetbrains.kotlin:kotlin-gradle-plugin:${kotlinVersion}") implementation("org.jetbrains.kotlin:kotlin-compiler-embeddable:${kotlinVersion}") diff --git a/buildSrc/src/main/java/org/springframework/boot/build/AntoraConventions.java b/buildSrc/src/main/java/org/springframework/boot/build/AntoraConventions.java index 1831393168ee..697eed64d328 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/AntoraConventions.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/AntoraConventions.java @@ -35,7 +35,9 @@ import org.antora.gradle.AntoraTask; import org.gradle.StartParameter; import org.gradle.api.Project; +import org.gradle.api.artifacts.Configuration; import org.gradle.api.file.Directory; +import org.gradle.api.file.FileCollection; import org.gradle.api.logging.LogLevel; import org.gradle.api.plugins.JavaBasePlugin; import org.gradle.api.provider.Provider; @@ -46,7 +48,7 @@ import org.springframework.boot.build.antora.AntoraAsciidocAttributes; import org.springframework.boot.build.antora.GenerateAntoraPlaybook; import org.springframework.boot.build.bom.BomExtension; -import org.springframework.boot.build.constraints.ExtractVersionConstraints; +import org.springframework.boot.build.bom.ResolvedBom; import org.springframework.util.Assert; import org.springframework.util.StringUtils; @@ -76,7 +78,10 @@ void apply(Project project) { } private void apply(Project project, AntoraPlugin antoraPlugin) { - TaskProvider dependencyVersionsTask = addDependencyVersionsTask(project); + Configuration resolvedBom = project.getConfigurations().create("resolveBom"); + project.getDependencies() + .add(resolvedBom.getName(), project.getDependencies() + .project(Map.of("path", DEPENDENCIES_PATH, "configuration", "resolvedBom"))); project.getPlugins().apply(GenerateAntoraYmlPlugin.class); TaskContainer tasks = project.getTasks(); TaskProvider generateAntoraPlaybookTask = tasks.register( @@ -86,8 +91,8 @@ private void apply(Project project, AntoraPlugin antoraPlugin) { (task) -> configureCopyAntoraPackageJsonTask(project, task)); TaskProvider npmInstallTask = tasks.register("antoraNpmInstall", NpmInstallTask.class, (task) -> configureNpmInstallTask(project, task, copyAntoraPackageJsonTask)); - tasks.withType(GenerateAntoraYmlTask.class, (generateAntoraYmlTask) -> configureGenerateAntoraYmlTask(project, - generateAntoraYmlTask, dependencyVersionsTask)); + tasks.withType(GenerateAntoraYmlTask.class, + (generateAntoraYmlTask) -> configureGenerateAntoraYmlTask(project, generateAntoraYmlTask, resolvedBom)); tasks.withType(AntoraTask.class, (antoraTask) -> configureAntoraTask(project, antoraTask, npmInstallTask, generateAntoraPlaybookTask)); project.getExtensions() @@ -118,21 +123,15 @@ private void configureNpmInstallTask(Project project, NpmInstallTask npmInstallT npmInstallTask.getNpmCommand().set(List.of("ci", "--silent", "--no-progress")); } - private TaskProvider addDependencyVersionsTask(Project project) { - return project.getTasks() - .register("dependencyVersions", ExtractVersionConstraints.class, - (task) -> task.enforcedPlatform(DEPENDENCIES_PATH)); - } - private void configureGenerateAntoraYmlTask(Project project, GenerateAntoraYmlTask generateAntoraYmlTask, - TaskProvider dependencyVersionsTask) { + Configuration resolvedBom) { generateAntoraYmlTask.getOutputs().doNotCacheIf("getAsciidocAttributes() changes output", (task) -> true); - generateAntoraYmlTask.dependsOn(dependencyVersionsTask); + generateAntoraYmlTask.dependsOn(resolvedBom); generateAntoraYmlTask.setProperty("componentName", "boot"); generateAntoraYmlTask.setProperty("outputFile", project.getLayout().getBuildDirectory().file("generated/docs/antora-yml/antora.yml")); generateAntoraYmlTask.setProperty("yml", getDefaultYml(project)); - generateAntoraYmlTask.getAsciidocAttributes().putAll(getAsciidocAttributes(project, dependencyVersionsTask)); + generateAntoraYmlTask.getAsciidocAttributes().putAll(getAsciidocAttributes(project, resolvedBom)); } private Map getDefaultYml(Project project) { @@ -151,11 +150,11 @@ private void configureGenerateAntoraYmlTask(Project project, GenerateAntoraYmlTa return defaultYml; } - private Provider> getAsciidocAttributes(Project project, - TaskProvider dependencyVersionsTask) { - return dependencyVersionsTask.map((task) -> task.getVersionConstraints()).map((constraints) -> { + private Provider> getAsciidocAttributes(Project project, FileCollection resolvedBoms) { + return project.provider(() -> { BomExtension bom = (BomExtension) project.project(DEPENDENCIES_PATH).getExtensions().getByName("bom"); - return new AntoraAsciidocAttributes(project, bom, constraints).get(); + ResolvedBom resolvedBom = ResolvedBom.readFrom(resolvedBoms.getSingleFile()); + return new AntoraAsciidocAttributes(project, bom, resolvedBom).get(); }); } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/MavenPublishingConventions.java b/buildSrc/src/main/java/org/springframework/boot/build/MavenPublishingConventions.java index 8bc2fb6efdb0..28fd088bda38 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/MavenPublishingConventions.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/MavenPublishingConventions.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,8 +16,8 @@ package org.springframework.boot.build; -import org.apache.maven.artifact.repository.MavenArtifactRepository; import org.gradle.api.Project; +import org.gradle.api.artifacts.repositories.MavenArtifactRepository; import org.gradle.api.attributes.Usage; import org.gradle.api.plugins.JavaPlugin; import org.gradle.api.plugins.JavaPluginExtension; diff --git a/buildSrc/src/main/java/org/springframework/boot/build/antora/AntoraAsciidocAttributes.java b/buildSrc/src/main/java/org/springframework/boot/build/antora/AntoraAsciidocAttributes.java index d32004c8794a..b5b19a4d57df 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/antora/AntoraAsciidocAttributes.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/antora/AntoraAsciidocAttributes.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,7 +20,9 @@ import java.io.InputStream; import java.io.UncheckedIOException; import java.util.Arrays; +import java.util.Collection; import java.util.Collections; +import java.util.HashMap; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; @@ -32,6 +34,10 @@ import org.springframework.boot.build.artifacts.ArtifactRelease; import org.springframework.boot.build.bom.BomExtension; import org.springframework.boot.build.bom.Library; +import org.springframework.boot.build.bom.ResolvedBom; +import org.springframework.boot.build.bom.ResolvedBom.Bom; +import org.springframework.boot.build.bom.ResolvedBom.Id; +import org.springframework.boot.build.bom.ResolvedBom.ResolvedLibrary; import org.springframework.boot.build.properties.BuildProperties; import org.springframework.boot.build.properties.BuildType; import org.springframework.util.Assert; @@ -59,17 +65,48 @@ public class AntoraAsciidocAttributes { private final Map projectProperties; - public AntoraAsciidocAttributes(Project project, BomExtension dependencyBom, - Map dependencyVersions) { + public AntoraAsciidocAttributes(Project project, BomExtension dependencyBom, ResolvedBom resolvedBom) { this.version = String.valueOf(project.getVersion()); this.latestVersion = Boolean.parseBoolean(String.valueOf(project.findProperty("latestVersion"))); this.buildType = BuildProperties.get(project).buildType(); this.artifactRelease = ArtifactRelease.forProject(project); this.libraries = dependencyBom.getLibraries(); - this.dependencyVersions = dependencyVersions; + this.dependencyVersions = dependencyVersionsOf(resolvedBom); this.projectProperties = project.getProperties(); } + private static Map dependencyVersionsOf(ResolvedBom resolvedBom) { + Map dependencyVersions = new HashMap<>(); + for (ResolvedLibrary library : resolvedBom.libraries()) { + dependencyVersions.putAll(dependencyVersionsOf(library.managedDependencies())); + for (Bom importedBom : library.importedBoms()) { + dependencyVersions.putAll(dependencyVersionsOf(importedBom)); + } + } + return dependencyVersions; + } + + private static Map dependencyVersionsOf(Bom bom) { + Map dependencyVersions = new HashMap<>(); + if (bom != null) { + dependencyVersions.putAll(dependencyVersionsOf(bom.managedDependencies())); + dependencyVersions.putAll(dependencyVersionsOf(bom.parent())); + for (Bom importedBom : bom.importedBoms()) { + dependencyVersions.putAll(dependencyVersionsOf(importedBom)); + } + } + return dependencyVersions; + } + + private static Map dependencyVersionsOf(Collection managedDependencies) { + Map dependencyVersions = new HashMap<>(); + for (Id managedDependency : managedDependencies) { + dependencyVersions.put(managedDependency.groupId() + ":" + managedDependency.artifactId(), + managedDependency.version()); + } + return dependencyVersions; + } + AntoraAsciidocAttributes(String version, boolean latestVersion, BuildType buildType, List libraries, Map dependencyVersions, Map projectProperties) { this.version = version; diff --git a/buildSrc/src/main/java/org/springframework/boot/build/architecture/ArchitectureRules.java b/buildSrc/src/main/java/org/springframework/boot/build/architecture/ArchitectureRules.java index 387b76acc463..6fb0390036b5 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/architecture/ArchitectureRules.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/architecture/ArchitectureRules.java @@ -23,6 +23,7 @@ import java.util.Map; import java.util.Objects; import java.util.function.BiConsumer; +import java.util.function.Predicate; import java.util.function.Supplier; import java.util.stream.Collectors; @@ -33,9 +34,11 @@ import com.tngtech.archunit.core.domain.JavaClass; import com.tngtech.archunit.core.domain.JavaClass.Predicates; import com.tngtech.archunit.core.domain.JavaMethod; +import com.tngtech.archunit.core.domain.JavaModifier; import com.tngtech.archunit.core.domain.JavaParameter; import com.tngtech.archunit.core.domain.JavaType; import com.tngtech.archunit.core.domain.properties.CanBeAnnotated; +import com.tngtech.archunit.core.domain.properties.HasAnnotations; import com.tngtech.archunit.core.domain.properties.HasName; import com.tngtech.archunit.core.domain.properties.HasOwner; import com.tngtech.archunit.core.domain.properties.HasOwner.Predicates.With; @@ -49,6 +52,8 @@ import com.tngtech.archunit.lang.syntax.elements.GivenMethodsConjunction; import com.tngtech.archunit.library.dependencies.SlicesRuleDefinition; +import org.springframework.beans.factory.config.BeanDefinition; +import org.springframework.context.annotation.Role; import org.springframework.util.ResourceUtils; /** @@ -88,6 +93,9 @@ static List standard() { rules.add(noClassesShouldCallStringToLowerCaseWithoutLocale()); rules.add(conditionalOnMissingBeanShouldNotSpecifyOnlyATypeThatIsTheSameAsMethodReturnType()); rules.add(enumSourceShouldNotSpecifyOnlyATypeThatIsTheSameAsMethodParameterType()); + rules.add(classLevelConfigurationPropertiesShouldNotSpecifyOnlyPrefixAttribute()); + rules.add(methodLevelConfigurationPropertiesShouldNotSpecifyOnlyPrefixAttribute()); + rules.add(conditionsShouldNotBePublic()); return List.copyOf(rules); } @@ -115,7 +123,8 @@ private static void allBeanPostProcessorBeanMethodsShouldBeStaticAndNotCausePrem .not(CanBeAnnotated.Predicates.annotatedWith("org.springframework.context.annotation.Lazy")); DescribedPredicate notOfASafeType = notAssignableTo( "org.springframework.beans.factory.ObjectProvider", "org.springframework.context.ApplicationContext", - "org.springframework.core.env.Environment"); + "org.springframework.core.env.Environment") + .and(notAnnotatedWithRoleInfrastructure()); item.getParameters() .stream() .filter(notAnnotatedWithLazy) @@ -125,6 +134,13 @@ private static void allBeanPostProcessorBeanMethodsShouldBeStaticAndNotCausePrem + notAnnotatedWithLazy.getDescription() + " and is " + notOfASafeType.getDescription())); } + private static DescribedPredicate notAnnotatedWithRoleInfrastructure() { + return is("not annotated with @Role(BeanDefinition.ROLE_INFRASTRUCTURE", (candidate) -> { + Role role = candidate.getAnnotationOfType(Role.class); + return (role == null) || (role.value() != BeanDefinition.ROLE_INFRASTRUCTURE); + }); + } + private static ArchRule allBeanFactoryPostProcessorBeanMethodsShouldBeStaticAndHaveOnlyInjectEnvironment() { return methodsThatAreAnnotatedWith("org.springframework.context.annotation.Bean").and() .haveRawReturnType(assignableTo("org.springframework.beans.factory.config.BeanFactoryPostProcessor")) @@ -244,6 +260,52 @@ private static void notSpecifyOnlyATypeThatIsTheSameAsTheMethodParameterType(Jav } } + private static ArchRule classLevelConfigurationPropertiesShouldNotSpecifyOnlyPrefixAttribute() { + return ArchRuleDefinition.classes() + .that() + .areAnnotatedWith("org.springframework.boot.context.properties.ConfigurationProperties") + .should(notSpecifyOnlyPrefixAttributeOfConfigurationProperties()) + .allowEmptyShould(true); + } + + private static ArchRule methodLevelConfigurationPropertiesShouldNotSpecifyOnlyPrefixAttribute() { + return ArchRuleDefinition.methods() + .that() + .areAnnotatedWith("org.springframework.boot.context.properties.ConfigurationProperties") + .should(notSpecifyOnlyPrefixAttributeOfConfigurationProperties()) + .allowEmptyShould(true); + } + + private static ArchCondition> notSpecifyOnlyPrefixAttributeOfConfigurationProperties() { + return check("not specify only prefix attribute of @ConfigurationProperties", + ArchitectureRules::notSpecifyOnlyPrefixAttributeOfConfigurationProperties); + } + + private static void notSpecifyOnlyPrefixAttributeOfConfigurationProperties(HasAnnotations item, + ConditionEvents events) { + JavaAnnotation configurationPropertiesAnnotation = item + .getAnnotationOfType("org.springframework.boot.context.properties.ConfigurationProperties"); + Map properties = configurationPropertiesAnnotation.getProperties(); + if (properties.size() == 1 && properties.containsKey("prefix")) { + addViolation(events, item, configurationPropertiesAnnotation.getDescription() + + " should specify implicit 'value' attribute other than explicit 'prefix' attribute"); + } + } + + private static ArchRule conditionsShouldNotBePublic() { + String springBootCondition = "org.springframework.boot.autoconfigure.condition.SpringBootCondition"; + return ArchRuleDefinition.noClasses() + .that() + .areAssignableTo(springBootCondition) + .and() + .doNotHaveModifier(JavaModifier.ABSTRACT) + .and() + .areNotAnnotatedWith(Deprecated.class) + .should() + .bePublic() + .allowEmptyShould(true); + } + private static boolean containsOnlySingleType(JavaType[] types, JavaType type) { return types.length == 1 && type.equals(types[0]); } @@ -286,6 +348,17 @@ private static DescribedPredicate assignableTo(String... typeNames) { return result; } + private static DescribedPredicate is(String description, Predicate predicate) { + return new DescribedPredicate<>(description) { + + @Override + public boolean test(JavaClass t) { + return predicate.test(t); + } + + }; + } + private static ArchCondition check(String description, BiConsumer check) { return new ArchCondition<>(description) { diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/BomExtension.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/BomExtension.java index 0c8804315286..114b19280f51 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/BomExtension.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/BomExtension.java @@ -16,10 +16,6 @@ package org.springframework.boot.build.bom; -import java.io.File; -import java.io.IOException; -import java.io.InputStreamReader; -import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.Collection; import java.util.HashMap; @@ -29,48 +25,30 @@ import java.util.function.Function; import javax.inject.Inject; -import javax.xml.parsers.DocumentBuilderFactory; -import javax.xml.transform.TransformerFactory; -import javax.xml.transform.dom.DOMSource; -import javax.xml.transform.stream.StreamResult; -import javax.xml.xpath.XPath; -import javax.xml.xpath.XPathConstants; -import javax.xml.xpath.XPathFactory; import groovy.lang.Closure; import groovy.lang.GroovyObjectSupport; import org.apache.maven.artifact.versioning.InvalidVersionSpecificationException; import org.apache.maven.artifact.versioning.VersionRange; import org.gradle.api.Action; -import org.gradle.api.GradleException; import org.gradle.api.InvalidUserCodeException; import org.gradle.api.InvalidUserDataException; import org.gradle.api.Project; -import org.gradle.api.Task; -import org.gradle.api.artifacts.Configuration; import org.gradle.api.artifacts.dsl.DependencyHandler; import org.gradle.api.model.ObjectFactory; import org.gradle.api.plugins.JavaPlatformPlugin; -import org.gradle.api.publish.maven.tasks.GenerateMavenPom; -import org.gradle.api.tasks.Sync; -import org.gradle.api.tasks.TaskExecutionException; -import org.gradle.api.tasks.TaskProvider; -import org.w3c.dom.Document; -import org.w3c.dom.NodeList; - -import org.springframework.boot.build.DeployedPlugin; -import org.springframework.boot.build.RepositoryTransformersExtension; + import org.springframework.boot.build.bom.Library.Exclusion; import org.springframework.boot.build.bom.Library.Group; +import org.springframework.boot.build.bom.Library.ImportedBom; import org.springframework.boot.build.bom.Library.LibraryVersion; import org.springframework.boot.build.bom.Library.Link; import org.springframework.boot.build.bom.Library.Module; +import org.springframework.boot.build.bom.Library.PermittedDependency; import org.springframework.boot.build.bom.Library.ProhibitedVersion; import org.springframework.boot.build.bom.Library.VersionAlignment; import org.springframework.boot.build.bom.bomr.version.DependencyVersion; -import org.springframework.boot.build.mavenplugin.MavenExec; import org.springframework.boot.build.properties.BuildProperties; -import org.springframework.util.FileCopyUtils; import org.springframework.util.PropertyPlaceholderHelper; import org.springframework.util.PropertyPlaceholderHelper.PlaceholderResolver; @@ -82,6 +60,8 @@ */ public class BomExtension { + private final String id; + private final Project project; private final UpgradeHandler upgradeHandler; @@ -95,6 +75,11 @@ public class BomExtension { public BomExtension(Project project) { this.project = project; this.upgradeHandler = project.getObjects().newInstance(UpgradeHandler.class, project); + this.id = "%s:%s:%s".formatted(project.getGroup(), project.getName(), project.getVersion()); + } + + public String getId() { + return this.id; } public List getLibraries() { @@ -131,65 +116,6 @@ public void library(String name, String version, Action action) libraryHandler.links)); } - public void effectiveBomArtifact() { - Configuration effectiveBomConfiguration = this.project.getConfigurations().create("effectiveBom"); - RepositoryTransformersExtension repositoryTransformers = this.project.getExtensions() - .getByType(RepositoryTransformersExtension.class); - this.project.getTasks() - .matching((task) -> task.getName().equals(DeployedPlugin.GENERATE_POM_TASK_NAME)) - .all((task) -> { - File generatedBomDir = this.project.getLayout() - .getBuildDirectory() - .dir("generated/bom") - .get() - .getAsFile(); - TaskProvider syncBom = this.project.getTasks().register("syncBom", Sync.class, (sync) -> { - sync.dependsOn(task); - sync.setDestinationDir(generatedBomDir); - sync.from(((GenerateMavenPom) task).getDestination(), (pom) -> pom.rename((name) -> "pom.xml")); - sync.from(this.project.getResources().getText().fromString(loadSettingsXml()), (settingsXml) -> { - settingsXml.rename((name) -> "settings.xml"); - settingsXml.filter(repositoryTransformers.mavenSettings()); - }); - }); - File effectiveBom = this.project.getLayout() - .getBuildDirectory() - .file("generated/effective-bom/" + this.project.getName() + "-effective-bom.xml") - .get() - .getAsFile(); - TaskProvider generateEffectiveBom = this.project.getTasks() - .register("generateEffectiveBom", MavenExec.class, (maven) -> { - maven.getProjectDir().set(generatedBomDir); - maven.args("--settings", "settings.xml", "help:effective-pom", "-Doutput=" + effectiveBom); - maven.dependsOn(syncBom); - maven.getOutputs().file(effectiveBom); - maven.doLast(new StripUnrepeatableOutputAction(effectiveBom)); - }); - this.project.getArtifacts() - .add(effectiveBomConfiguration.getName(), effectiveBom, - (artifact) -> artifact.builtBy(generateEffectiveBom)); - }); - } - - private String loadSettingsXml() { - try { - return FileCopyUtils - .copyToString(new InputStreamReader( - getClass().getClassLoader().getResourceAsStream("effective-bom-settings.xml"), - StandardCharsets.UTF_8)) - .replace("localRepositoryPath", - this.project.getLayout() - .getBuildDirectory() - .dir("local-m2-repository") - .get() - .getAsFile() - .getAbsolutePath()); - } - catch (IOException ex) { - throw new GradleException("Failed to prepare settings.xml", ex); - } - } - private String createDependencyNotation(String groupId, String artifactId, DependencyVersion version) { return groupId + ":" + artifactId + ":" + version; } @@ -228,8 +154,8 @@ private void addLibrary(Library library) { for (Module module : group.getModules()) { addModule(library, dependencies, versionProperty, group, module); } - for (String bomImport : group.getBoms()) { - addBomImport(library, dependencies, versionProperty, group, bomImport); + for (ImportedBom bomImport : group.getBoms()) { + addBomImport(library, dependencies, versionProperty, group, bomImport.name()); } } } @@ -252,6 +178,8 @@ private void addBomImport(Library library, DependencyHandler dependencies, Strin public static class LibraryHandler { + private final Project project; + private final List groups = new ArrayList<>(); private final List prohibitedVersions = new ArrayList<>(); @@ -270,6 +198,7 @@ public static class LibraryHandler { @Inject public LibraryHandler(Project project, String version) { + this.project = project; this.version = version; this.alignWith = project.getObjects().newInstance(AlignWithHandler.class); } @@ -287,7 +216,7 @@ public void setCalendarName(String calendarName) { } public void group(String id, Action action) { - GroupHandler groupHandler = new GroupHandler(id); + GroupHandler groupHandler = this.project.getObjects().newInstance(GroupHandler.class, id); action.execute(groupHandler); this.groups .add(new Group(groupHandler.id, groupHandler.modules, groupHandler.plugins, groupHandler.imports)); @@ -366,16 +295,17 @@ public void because(String because) { } - public class GroupHandler extends GroovyObjectSupport { + public static class GroupHandler extends GroovyObjectSupport { private final String id; private List modules = new ArrayList<>(); - private List imports = new ArrayList<>(); + private List imports = new ArrayList<>(); private List plugins = new ArrayList<>(); + @Inject public GroupHandler(String id) { this.id = id; } @@ -386,8 +316,14 @@ public void setModules(List modules) { .toList(); } - public void setImports(List imports) { - this.imports = imports; + public void bom(String bom) { + this.imports.add(new ImportedBom(bom)); + } + + public void bom(String bom, Action action) { + ImportBomHandler handler = new ImportBomHandler(); + action.execute(handler); + this.imports.add(new ImportedBom(bom, handler.permittedDependencies)); } public void setPlugins(List plugins) { @@ -429,6 +365,17 @@ public void setClassifier(String classifier) { } + public class ImportBomHandler { + + private final List permittedDependencies = new ArrayList<>(); + + public void permit(String allowed) { + String[] components = allowed.split(":"); + this.permittedDependencies.add(new PermittedDependency(components[0], components[1])); + } + + } + } public static class AlignWithHandler { @@ -647,39 +594,4 @@ public List getIssueLabels() { } - private static final class StripUnrepeatableOutputAction implements Action { - - private final File effectiveBom; - - private StripUnrepeatableOutputAction(File xmlFile) { - this.effectiveBom = xmlFile; - } - - @Override - public void execute(Task task) { - try { - Document document = DocumentBuilderFactory.newInstance().newDocumentBuilder().parse(this.effectiveBom); - XPath xpath = XPathFactory.newInstance().newXPath(); - NodeList comments = (NodeList) xpath.evaluate("//comment()", document, XPathConstants.NODESET); - for (int i = 0; i < comments.getLength(); i++) { - org.w3c.dom.Node comment = comments.item(i); - comment.getParentNode().removeChild(comment); - } - org.w3c.dom.Node build = (org.w3c.dom.Node) xpath.evaluate("/project/build", document, - XPathConstants.NODE); - build.getParentNode().removeChild(build); - org.w3c.dom.Node reporting = (org.w3c.dom.Node) xpath.evaluate("/project/reporting", document, - XPathConstants.NODE); - reporting.getParentNode().removeChild(reporting); - TransformerFactory.newInstance() - .newTransformer() - .transform(new DOMSource(document), new StreamResult(this.effectiveBom)); - } - catch (Exception ex) { - throw new TaskExecutionException(task, ex); - } - } - - } - } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/BomPlugin.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/BomPlugin.java index edb415458241..bb100034e16f 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/BomPlugin.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/BomPlugin.java @@ -62,11 +62,19 @@ public void apply(Project project) { javaPlatform.allowDependencies(); createApiEnforcedConfiguration(project); BomExtension bom = project.getExtensions().create("bom", BomExtension.class, project); + TaskProvider createResolvedBom = project.getTasks() + .register("createResolvedBom", CreateResolvedBom.class, bom); TaskProvider checkBom = project.getTasks().register("bomrCheck", CheckBom.class, bom); + checkBom.configure( + (task) -> task.getResolvedBomFile().set(createResolvedBom.flatMap(CreateResolvedBom::getOutputFile))); project.getTasks().named("check").configure((check) -> check.dependsOn(checkBom)); project.getTasks().register("bomrUpgrade", UpgradeBom.class, bom); project.getTasks().register("moveToSnapshots", MoveToSnapshots.class, bom); project.getTasks().register("checkLinks", CheckLinks.class, bom); + Configuration resolvedBomConfiguration = project.getConfigurations().create("resolvedBom"); + project.getArtifacts() + .add(resolvedBomConfiguration.getName(), createResolvedBom.map(CreateResolvedBom::getOutputFile), + (artifact) -> artifact.builtBy(createResolvedBom)); new PublishingCustomizer(project, bom).customize(); } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/BomResolver.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/BomResolver.java new file mode 100644 index 000000000000..ff02527086ee --- /dev/null +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/BomResolver.java @@ -0,0 +1,311 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.build.bom; + +import java.io.File; +import java.net.URI; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.function.Function; + +import javax.xml.namespace.QName; +import javax.xml.parsers.DocumentBuilder; +import javax.xml.parsers.DocumentBuilderFactory; +import javax.xml.parsers.ParserConfigurationException; +import javax.xml.xpath.XPath; +import javax.xml.xpath.XPathConstants; +import javax.xml.xpath.XPathExpressionException; +import javax.xml.xpath.XPathFactory; + +import org.gradle.api.artifacts.ConfigurationContainer; +import org.gradle.api.artifacts.ResolvedArtifact; +import org.gradle.api.artifacts.dsl.DependencyHandler; +import org.w3c.dom.Document; +import org.w3c.dom.NodeList; + +import org.springframework.boot.build.bom.Library.Group; +import org.springframework.boot.build.bom.Library.ImportedBom; +import org.springframework.boot.build.bom.Library.Link; +import org.springframework.boot.build.bom.Library.Module; +import org.springframework.boot.build.bom.ResolvedBom.Bom; +import org.springframework.boot.build.bom.ResolvedBom.Id; +import org.springframework.boot.build.bom.ResolvedBom.JavadocLink; +import org.springframework.boot.build.bom.ResolvedBom.Links; +import org.springframework.boot.build.bom.ResolvedBom.ResolvedLibrary; + +/** + * Creates a {@link ResolvedBom resolved bom}. + * + * @author Andy Wilkinson + */ +class BomResolver { + + private final ConfigurationContainer configurations; + + private final DependencyHandler dependencies; + + private final DocumentBuilder documentBuilder; + + BomResolver(ConfigurationContainer configurations, DependencyHandler dependencies) { + this.configurations = configurations; + this.dependencies = dependencies; + try { + this.documentBuilder = DocumentBuilderFactory.newInstance().newDocumentBuilder(); + } + catch (ParserConfigurationException ex) { + throw new RuntimeException(ex); + } + } + + ResolvedBom resolve(BomExtension bomExtension) { + List libraries = new ArrayList<>(); + for (Library library : bomExtension.getLibraries()) { + List managedDependencies = new ArrayList<>(); + List imports = new ArrayList<>(); + for (Group group : library.getGroups()) { + for (Module module : group.getModules()) { + Id id = new Id(group.getId(), module.getName(), library.getVersion().getVersion().toString()); + managedDependencies.add(id); + } + for (ImportedBom imported : group.getBoms()) { + Bom bom = bomFrom(resolveBom( + "%s:%s:%s".formatted(group.getId(), imported.name(), library.getVersion().getVersion()))); + imports.add(bom); + } + } + List javadocLinks = javadocLinksOf(library).stream() + .map((link) -> new JavadocLink(URI.create(link.url(library)), link.packages())) + .toList(); + ResolvedLibrary resolvedLibrary = new ResolvedLibrary(library.getName(), + library.getVersion().getVersion().toString(), library.getVersionProperty(), managedDependencies, + imports, new Links(javadocLinks)); + libraries.add(resolvedLibrary); + } + String[] idComponents = bomExtension.getId().split(":"); + return new ResolvedBom(new Id(idComponents[0], idComponents[1], idComponents[2]), libraries); + } + + private List javadocLinksOf(Library library) { + List javadocLinks = library.getLinks("javadoc"); + return (javadocLinks != null) ? javadocLinks : Collections.emptyList(); + } + + Bom resolveMavenBom(String coordinates) { + return bomFrom(resolveBom(coordinates)); + } + + private File resolveBom(String coordinates) { + Set artifacts = this.configurations + .detachedConfiguration(this.dependencies.create(coordinates + "@pom")) + .getResolvedConfiguration() + .getResolvedArtifacts(); + if (artifacts.size() != 1) { + throw new IllegalStateException("Expected a single artifact but '%s' resolved to %d artifacts" + .formatted(coordinates, artifacts.size())); + } + return artifacts.iterator().next().getFile(); + } + + private Bom bomFrom(File bomFile) { + try { + Node bom = nodeFrom(bomFile); + File parentBomFile = parentBomFile(bom); + Bom parent = null; + if (parentBomFile != null) { + parent = bomFrom(parentBomFile); + } + Properties properties = Properties.from(bom, this::nodeFrom); + List dependencyNodes = bom.nodesAt("/project/dependencyManagement/dependencies/dependency"); + List managedDependencies = new ArrayList<>(); + List imports = new ArrayList<>(); + for (Node dependency : dependencyNodes) { + String groupId = properties.replace(dependency.textAt("groupId")); + String artifactId = properties.replace(dependency.textAt("artifactId")); + String version = properties.replace(dependency.textAt("version")); + String classifier = properties.replace(dependency.textAt("classifier")); + String scope = properties.replace(dependency.textAt("scope")); + Bom importedBom = null; + if ("import".equals(scope)) { + String type = properties.replace(dependency.textAt("type")); + if ("pom".equals(type)) { + importedBom = bomFrom(resolveBom(groupId + ":" + artifactId + ":" + version)); + } + } + if (importedBom != null) { + imports.add(importedBom); + } + else { + managedDependencies.add(new Id(groupId, artifactId, version, classifier)); + } + } + String groupId = bom.textAt("/project/groupId"); + if ((groupId == null || groupId.isEmpty()) && parent != null) { + groupId = parent.id().groupId(); + } + String artifactId = bom.textAt("/project/artifactId"); + String version = bom.textAt("/project/version"); + if ((version == null || version.isEmpty()) && parent != null) { + version = parent.id().version(); + } + return new Bom(new Id(groupId, artifactId, version), parent, managedDependencies, imports); + } + catch (Exception ex) { + throw new RuntimeException(ex); + } + } + + private Node nodeFrom(String coordinates) { + return nodeFrom(resolveBom(coordinates)); + } + + private Node nodeFrom(File bomFile) { + try { + Document document = this.documentBuilder.parse(bomFile); + return new Node(document); + } + catch (Exception ex) { + throw new RuntimeException(ex); + } + } + + private File parentBomFile(Node bom) { + Node parent = bom.nodeAt("/project/parent"); + if (parent != null) { + String parentGroupId = parent.textAt("groupId"); + String parentArtifactId = parent.textAt("artifactId"); + String parentVersion = parent.textAt("version"); + return resolveBom(parentGroupId + ":" + parentArtifactId + ":" + parentVersion); + } + return null; + } + + private static final class Node { + + protected final XPath xpath; + + private final org.w3c.dom.Node delegate; + + private Node(org.w3c.dom.Node delegate) { + this(delegate, XPathFactory.newInstance().newXPath()); + } + + private Node(org.w3c.dom.Node delegate, XPath xpath) { + this.delegate = delegate; + this.xpath = xpath; + } + + private String textAt(String expression) { + String text = (String) evaluate(expression + "/text()", XPathConstants.STRING); + return (text != null && !text.isBlank()) ? text : null; + } + + private Node nodeAt(String expression) { + org.w3c.dom.Node result = (org.w3c.dom.Node) evaluate(expression, XPathConstants.NODE); + return (result != null) ? new Node(result, this.xpath) : null; + } + + private List nodesAt(String expression) { + NodeList nodes = (NodeList) evaluate(expression, XPathConstants.NODESET); + List things = new ArrayList<>(nodes.getLength()); + for (int i = 0; i < nodes.getLength(); i++) { + things.add(new Node(nodes.item(i), this.xpath)); + } + return things; + } + + private Object evaluate(String expression, QName type) { + try { + return this.xpath.evaluate(expression, this.delegate, type); + } + catch (XPathExpressionException ex) { + throw new RuntimeException(ex); + } + } + + private String name() { + return this.delegate.getNodeName(); + } + + private String textContent() { + return this.delegate.getTextContent(); + } + + } + + private static final class Properties { + + private final Map properties; + + private Properties(Map properties) { + this.properties = properties; + } + + private static Properties from(Node bom, Function resolver) { + try { + Map properties = new HashMap<>(); + Node current = bom; + while (current != null) { + String groupId = current.textAt("/project/groupId"); + if (groupId != null && !groupId.isEmpty()) { + properties.putIfAbsent("${project.groupId}", groupId); + } + String version = current.textAt("/project/version"); + if (version != null && !version.isEmpty()) { + properties.putIfAbsent("${project.version}", version); + } + List propertyNodes = current.nodesAt("/project/properties/*"); + for (Node property : propertyNodes) { + properties.putIfAbsent("${%s}".formatted(property.name()), property.textContent()); + } + current = parent(current, resolver); + } + return new Properties(properties); + } + catch (Exception ex) { + throw new RuntimeException(ex); + } + } + + private static Node parent(Node current, Function resolver) { + Node parent = current.nodeAt("/project/parent"); + if (parent != null) { + String parentGroupId = parent.textAt("groupId"); + String parentArtifactId = parent.textAt("artifactId"); + String parentVersion = parent.textAt("version"); + return resolver.apply(parentGroupId + ":" + parentArtifactId + ":" + parentVersion); + } + return null; + } + + private String replace(String input) { + if (input != null && input.startsWith("${") && input.endsWith("}")) { + String value = this.properties.get(input); + if (value != null) { + return replace(value); + } + throw new IllegalStateException("No replacement for " + input); + } + return input; + } + + } + +} diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/CheckBom.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/CheckBom.java index 365e67c6784c..585fd149fb05 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/CheckBom.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/CheckBom.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,9 +16,13 @@ package org.springframework.boot.build.bom; -import java.io.File; import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.LinkedHashMap; import java.util.List; +import java.util.Map; +import java.util.Optional; import java.util.Set; import java.util.TreeSet; import java.util.stream.Collectors; @@ -32,15 +36,24 @@ import org.gradle.api.DefaultTask; import org.gradle.api.GradleException; import org.gradle.api.artifacts.ConfigurationContainer; -import org.gradle.api.artifacts.ResolvedArtifact; import org.gradle.api.artifacts.dsl.DependencyHandler; +import org.gradle.api.file.RegularFile; +import org.gradle.api.file.RegularFileProperty; +import org.gradle.api.provider.Provider; +import org.gradle.api.tasks.InputFile; +import org.gradle.api.tasks.PathSensitive; +import org.gradle.api.tasks.PathSensitivity; import org.gradle.api.tasks.TaskAction; import org.springframework.boot.build.bom.Library.Group; +import org.springframework.boot.build.bom.Library.ImportedBom; import org.springframework.boot.build.bom.Library.Module; +import org.springframework.boot.build.bom.Library.PermittedDependency; import org.springframework.boot.build.bom.Library.ProhibitedVersion; import org.springframework.boot.build.bom.Library.VersionAlignment; -import org.springframework.boot.build.bom.ManagedDependencies.Difference; +import org.springframework.boot.build.bom.ResolvedBom.Bom; +import org.springframework.boot.build.bom.ResolvedBom.Id; +import org.springframework.boot.build.bom.ResolvedBom.ResolvedLibrary; import org.springframework.boot.build.bom.bomr.version.DependencyVersion; /** @@ -51,24 +64,31 @@ */ public abstract class CheckBom extends DefaultTask { - private final ConfigurationContainer configurations; - - private final DependencyHandler dependencies; - private final BomExtension bom; + private final List checks; + @Inject public CheckBom(BomExtension bom) { + ConfigurationContainer configurations = getProject().getConfigurations(); + DependencyHandler dependencies = getProject().getDependencies(); + Provider resolvedBom = getResolvedBomFile().map(RegularFile::getAsFile).map(ResolvedBom::readFrom); + this.checks = List.of(new CheckExclusions(configurations, dependencies), new CheckProhibitedVersions(), + new CheckVersionAlignment(), + new CheckDependencyManagementAlignment(resolvedBom, configurations, dependencies), + new CheckForUnwantedDependencyManagement(resolvedBom)); this.bom = bom; - this.configurations = getProject().getConfigurations(); - this.dependencies = getProject().getDependencies(); } + @InputFile + @PathSensitive(PathSensitivity.RELATIVE) + public abstract RegularFileProperty getResolvedBomFile(); + @TaskAction void checkBom() { List errors = new ArrayList<>(); for (Library library : this.bom.getLibraries()) { - checkLibrary(library, errors); + errors.addAll(checkLibrary(library)); } if (!errors.isEmpty()) { System.out.println(); @@ -78,148 +98,348 @@ void checkBom() { } } - private void checkLibrary(Library library, List errors) { + private List checkLibrary(Library library) { List libraryErrors = new ArrayList<>(); - checkExclusions(library, libraryErrors); - checkProhibitedVersions(library, libraryErrors); - checkVersionAlignment(library, libraryErrors); - checkDependencyManagementAlignment(library, libraryErrors); + this.checks.stream().flatMap((check) -> check.check(library).stream()).forEach(libraryErrors::add); + List errors = new ArrayList<>(); if (!libraryErrors.isEmpty()) { errors.add(library.getName()); for (String libraryError : libraryErrors) { errors.add(" - " + libraryError); } } + return errors; } - private void checkExclusions(Library library, List errors) { - for (Group group : library.getGroups()) { - for (Module module : group.getModules()) { - if (!module.getExclusions().isEmpty()) { - checkExclusions(group.getId(), module, library.getVersion().getVersion(), errors); + private interface LibraryCheck { + + List check(Library library); + + } + + private static final class CheckExclusions implements LibraryCheck { + + private final ConfigurationContainer configurations; + + private final DependencyHandler dependencies; + + private CheckExclusions(ConfigurationContainer configurations, DependencyHandler dependencies) { + this.configurations = configurations; + this.dependencies = dependencies; + } + + @Override + public List check(Library library) { + List errors = new ArrayList<>(); + for (Group group : library.getGroups()) { + for (Module module : group.getModules()) { + if (!module.getExclusions().isEmpty()) { + checkExclusions(group.getId(), module, library.getVersion().getVersion(), errors); + } } } + return errors; } - } - private void checkExclusions(String groupId, Module module, DependencyVersion version, List errors) { - Set resolved = this.configurations - .detachedConfiguration(this.dependencies.create(groupId + ":" + module.getName() + ":" + version)) - .getResolvedConfiguration() - .getResolvedArtifacts() - .stream() - .map((artifact) -> artifact.getModuleVersion().getId()) - .map((id) -> id.getGroup() + ":" + id.getModule().getName()) - .collect(Collectors.toSet()); - Set exclusions = module.getExclusions() - .stream() - .map((exclusion) -> exclusion.getGroupId() + ":" + exclusion.getArtifactId()) - .collect(Collectors.toSet()); - Set unused = new TreeSet<>(); - for (String exclusion : exclusions) { - if (!resolved.contains(exclusion)) { - if (exclusion.endsWith(":*")) { - String group = exclusion.substring(0, exclusion.indexOf(':') + 1); - if (resolved.stream().noneMatch((candidate) -> candidate.startsWith(group))) { + private void checkExclusions(String groupId, Module module, DependencyVersion version, List errors) { + Set resolved = this.configurations + .detachedConfiguration(this.dependencies.create(groupId + ":" + module.getName() + ":" + version)) + .getResolvedConfiguration() + .getResolvedArtifacts() + .stream() + .map((artifact) -> artifact.getModuleVersion().getId()) + .map((id) -> id.getGroup() + ":" + id.getModule().getName()) + .collect(Collectors.toSet()); + Set exclusions = module.getExclusions() + .stream() + .map((exclusion) -> exclusion.getGroupId() + ":" + exclusion.getArtifactId()) + .collect(Collectors.toSet()); + Set unused = new TreeSet<>(); + for (String exclusion : exclusions) { + if (!resolved.contains(exclusion)) { + if (exclusion.endsWith(":*")) { + String group = exclusion.substring(0, exclusion.indexOf(':') + 1); + if (resolved.stream().noneMatch((candidate) -> candidate.startsWith(group))) { + unused.add(exclusion); + } + } + else { unused.add(exclusion); } } + } + exclusions.removeAll(resolved); + if (!unused.isEmpty()) { + errors.add("Unnecessary exclusions on " + groupId + ":" + module.getName() + ": " + exclusions); + } + } + + } + + private static final class CheckProhibitedVersions implements LibraryCheck { + + @Override + public List check(Library library) { + List errors = new ArrayList<>(); + ArtifactVersion currentVersion = new DefaultArtifactVersion(library.getVersion().getVersion().toString()); + for (ProhibitedVersion prohibited : library.getProhibitedVersions()) { + if (prohibited.isProhibited(library.getVersion().getVersion().toString())) { + errors.add("Current version " + currentVersion + " is prohibited"); + } else { - unused.add(exclusion); + VersionRange versionRange = prohibited.getRange(); + if (versionRange != null) { + check(currentVersion, versionRange, errors); + } } } + return errors; } - exclusions.removeAll(resolved); - if (!unused.isEmpty()) { - errors.add("Unnecessary exclusions on " + groupId + ":" + module.getName() + ": " + exclusions); + + private void check(ArtifactVersion currentVersion, VersionRange versionRange, List errors) { + for (Restriction restriction : versionRange.getRestrictions()) { + ArtifactVersion upperBound = restriction.getUpperBound(); + if (upperBound == null) { + return; + } + int comparison = currentVersion.compareTo(upperBound); + if ((restriction.isUpperBoundInclusive() && comparison <= 0) + || ((!restriction.isUpperBoundInclusive()) && comparison < 0)) { + return; + } + } + errors.add("Version range " + versionRange + " is ineffective as the current version, " + currentVersion + + ", is greater than its upper bound"); } + } - private void checkProhibitedVersions(Library library, List errors) { - ArtifactVersion currentVersion = new DefaultArtifactVersion(library.getVersion().getVersion().toString()); - for (ProhibitedVersion prohibited : library.getProhibitedVersions()) { - if (prohibited.isProhibited(library.getVersion().getVersion().toString())) { - errors.add("Current version " + currentVersion + " is prohibited"); + private static final class CheckVersionAlignment implements LibraryCheck { + + @Override + public List check(Library library) { + List errors = new ArrayList<>(); + VersionAlignment versionAlignment = library.getVersionAlignment(); + if (versionAlignment != null) { + check(versionAlignment, library, errors); + } + return errors; + } + + private void check(VersionAlignment versionAlignment, Library library, List errors) { + Set alignedVersions = versionAlignment.resolve(); + if (alignedVersions.size() == 1) { + String alignedVersion = alignedVersions.iterator().next(); + if (!alignedVersion.equals(library.getVersion().getVersion().toString())) { + errors.add("Version " + library.getVersion().getVersion() + " is misaligned. It should be " + + alignedVersion + "."); + } } else { - VersionRange versionRange = prohibited.getRange(); - if (versionRange != null) { - for (Restriction restriction : versionRange.getRestrictions()) { - ArtifactVersion upperBound = restriction.getUpperBound(); - if (upperBound == null) { - return; - } - int comparison = currentVersion.compareTo(upperBound); - if ((restriction.isUpperBoundInclusive() && comparison <= 0) - || ((!restriction.isUpperBoundInclusive()) && comparison < 0)) { - return; - } - } - errors.add("Version range " + versionRange + " is ineffective as the current version, " - + currentVersion + ", is greater than its upper bound"); + if (alignedVersions.isEmpty()) { + errors.add("Version alignment requires a single version but none were found."); + } + else { + errors.add("Version alignment requires a single version but " + alignedVersions.size() + + " were found: " + alignedVersions + "."); } } } + } - private void checkVersionAlignment(Library library, List errors) { - VersionAlignment versionAlignment = library.getVersionAlignment(); - if (versionAlignment == null) { - return; + private abstract static class ResolvedLibraryCheck implements LibraryCheck { + + private final Provider resolvedBom; + + private ResolvedLibraryCheck(Provider resolvedBom) { + this.resolvedBom = resolvedBom; } - Set alignedVersions = versionAlignment.resolve(); - if (alignedVersions.size() == 1) { - String alignedVersion = alignedVersions.iterator().next(); - if (!alignedVersion.equals(library.getVersion().getVersion().toString())) { - errors.add("Version " + library.getVersion().getVersion() + " is misaligned. It should be " - + alignedVersion + "."); - } + + @Override + public List check(Library library) { + ResolvedLibrary resolvedLibrary = getResolvedLibrary(library); + return check(library, resolvedLibrary); } - else { - if (alignedVersions.isEmpty()) { - errors.add("Version alignment requires a single version but none were found."); - } - else { - errors.add("Version alignment requires a single version but " + alignedVersions.size() + " were found: " - + alignedVersions + "."); + + protected abstract List check(Library library, ResolvedLibrary resolvedLibrary); + + private ResolvedLibrary getResolvedLibrary(Library library) { + ResolvedBom resolvedBom = this.resolvedBom.get(); + Optional resolvedLibrary = resolvedBom.libraries() + .stream() + .filter((candidate) -> candidate.name().equals(library.getName())) + .findFirst(); + if (!resolvedLibrary.isPresent()) { + throw new RuntimeException("Library '%s' not found in resolved bom".formatted(library.getName())); } + return resolvedLibrary.get(); } + } - private void checkDependencyManagementAlignment(Library library, List errors) { - String alignsWithBom = library.getAlignsWithBom(); - if (alignsWithBom == null) { - return; - } - File bom = resolveBom(library, alignsWithBom); - ManagedDependencies managedByBom = ManagedDependencies.ofBom(bom); - ManagedDependencies managedByLibrary = ManagedDependencies.ofLibrary(library); - Difference diff = managedByBom.diff(managedByLibrary); - if (!diff.isEmpty()) { - String error = "Dependency management does not align with " + library.getAlignsWithBom() + ":"; - if (!diff.missing().isEmpty()) { - error = error + "%n - Missing:%n %s" - .formatted(String.join("\n ", diff.missing())); - } - if (!diff.unexpected().isEmpty()) { - error = error + "%n - Unexpected:%n %s" - .formatted(String.join("\n ", diff.unexpected())); + private static final class CheckDependencyManagementAlignment extends ResolvedLibraryCheck { + + private final BomResolver bomResolver; + + private CheckDependencyManagementAlignment(Provider resolvedBom, + ConfigurationContainer configurations, DependencyHandler dependencies) { + super(resolvedBom); + this.bomResolver = new BomResolver(configurations, dependencies); + } + + @Override + public List check(Library library, ResolvedLibrary resolvedLibrary) { + List errors = new ArrayList<>(); + String alignsWithBom = library.getAlignsWithBom(); + if (alignsWithBom != null) { + Bom mavenBom = this.bomResolver + .resolveMavenBom(alignsWithBom + ":" + library.getVersion().getVersion()); + checkDependencyManagementAlignment(resolvedLibrary, mavenBom, errors); + } + return errors; + } + + private void checkDependencyManagementAlignment(ResolvedLibrary library, Bom mavenBom, List errors) { + List managedByLibrary = library.managedDependencies(); + List managedByBom = managedDependenciesOf(mavenBom); + + List missing = new ArrayList<>(managedByBom); + missing.removeAll(managedByLibrary); + + List unexpected = new ArrayList<>(managedByLibrary); + unexpected.removeAll(managedByBom); + if (missing.isEmpty() && unexpected.isEmpty()) { + return; + } + String error = "Dependency management does not align with " + mavenBom.id() + ":"; + if (!missing.isEmpty()) { + error = error + "%n - Missing:%n %s".formatted(String.join("\n ", + missing.stream().map((dependency) -> dependency.toString()).toList())); + } + if (!unexpected.isEmpty()) { + error = error + "%n - Unexpected:%n %s".formatted(String.join("\n ", + unexpected.stream().map((dependency) -> dependency.toString()).toList())); } errors.add(error); } + + private List managedDependenciesOf(Bom mavenBom) { + List managedDependencies = new ArrayList<>(); + managedDependencies.addAll(mavenBom.managedDependencies()); + if (mavenBom.parent() != null) { + managedDependencies.addAll(managedDependenciesOf(mavenBom.parent())); + } + for (Bom importedBom : mavenBom.importedBoms()) { + managedDependencies.addAll(managedDependenciesOf(importedBom)); + } + return managedDependencies; + } + } - private File resolveBom(Library library, String alignsWithBom) { - String coordinates = alignsWithBom + ":" + library.getVersion().getVersion() + "@pom"; - Set artifacts = this.configurations - .detachedConfiguration(this.dependencies.create(coordinates)) - .getResolvedConfiguration() - .getResolvedArtifacts(); - if (artifacts.size() != 1) { - throw new IllegalStateException("Expected a single artifact but '%s' resolved to %d artifacts" - .formatted(coordinates, artifacts.size())); - } - return artifacts.iterator().next().getFile(); + private static final class CheckForUnwantedDependencyManagement extends ResolvedLibraryCheck { + + private CheckForUnwantedDependencyManagement(Provider resolvedBom) { + super(resolvedBom); + } + + @Override + public List check(Library library, ResolvedLibrary resolvedLibrary) { + Map> unwanted = findUnwantedDependencyManagement(library, resolvedLibrary); + List errors = new ArrayList<>(); + if (!unwanted.isEmpty()) { + StringBuilder error = new StringBuilder("Unwanted dependency management:"); + unwanted.forEach((bom, dependencies) -> { + error.append("%n - %s:".formatted(bom)); + error.append("%n - %s".formatted(String.join("\n - ", dependencies))); + }); + errors.add(error.toString()); + } + Map> unnecessary = findUnnecessaryPermittedDependencies(library, resolvedLibrary); + if (!unnecessary.isEmpty()) { + StringBuilder error = new StringBuilder("Dependencies permitted unnecessarily:"); + unnecessary.forEach((bom, dependencies) -> { + error.append("%n - %s:".formatted(bom)); + error.append("%n - %s".formatted(String.join("\n - ", dependencies))); + }); + errors.add(error.toString()); + } + return errors; + } + + private Map> findUnwantedDependencyManagement(Library library, + ResolvedLibrary resolvedLibrary) { + Map> unwanted = new LinkedHashMap<>(); + for (Bom bom : resolvedLibrary.importedBoms()) { + Set notPermitted = new TreeSet<>(); + Set managedDependencies = managedDependenciesOf(bom); + managedDependencies.stream() + .filter((dependency) -> unwanted(bom, dependency, findPermittedDependencies(library, bom))) + .map(Id::toString) + .forEach(notPermitted::add); + if (!notPermitted.isEmpty()) { + unwanted.put(bom.id().artifactId(), notPermitted); + } + } + return unwanted; + } + + private List findPermittedDependencies(Library library, Bom bom) { + for (Group group : library.getGroups()) { + for (ImportedBom importedBom : group.getBoms()) { + if (importedBom.name().equals(bom.id().artifactId()) && group.getId().equals(bom.id().groupId())) { + return importedBom.permittedDependencies(); + } + } + } + return Collections.emptyList(); + } + + private Set managedDependenciesOf(Bom bom) { + Set managedDependencies = new TreeSet<>(); + if (bom != null) { + managedDependencies.addAll(bom.managedDependencies()); + managedDependencies.addAll(managedDependenciesOf(bom.parent())); + for (Bom importedBom : bom.importedBoms()) { + managedDependencies.addAll(managedDependenciesOf(importedBom)); + } + } + return managedDependencies; + } + + private boolean unwanted(Bom bom, Id managedDependency, List permittedDependencies) { + if (bom.id().groupId().equals(managedDependency.groupId()) + || managedDependency.groupId().startsWith(bom.id().groupId() + ".")) { + return false; + } + for (PermittedDependency permittedDependency : permittedDependencies) { + if (permittedDependency.artifactId().equals(managedDependency.artifactId()) + && permittedDependency.groupId().equals(managedDependency.groupId())) { + return false; + } + } + return true; + } + + private Map> findUnnecessaryPermittedDependencies(Library library, + ResolvedLibrary resolvedLibrary) { + Map> unnecessary = new HashMap<>(); + for (Bom bom : resolvedLibrary.importedBoms()) { + Set permittedDependencies = findPermittedDependencies(library, bom).stream() + .map((dependency) -> dependency.groupId() + ":" + dependency.artifactId()) + .collect(Collectors.toCollection(TreeSet::new)); + Set dependencies = managedDependenciesOf(bom).stream() + .map((dependency) -> dependency.groupId() + ":" + dependency.artifactId()) + .collect(Collectors.toCollection(TreeSet::new)); + permittedDependencies.removeAll(dependencies); + if (!permittedDependencies.isEmpty()) { + unnecessary.put(bom.id().artifactId(), permittedDependencies); + } + } + return unnecessary; + } + } } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/CreateResolvedBom.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/CreateResolvedBom.java new file mode 100644 index 000000000000..f84c69ffa929 --- /dev/null +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/CreateResolvedBom.java @@ -0,0 +1,60 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.build.bom; + +import java.io.FileWriter; +import java.io.IOException; + +import javax.inject.Inject; + +import org.gradle.api.DefaultTask; +import org.gradle.api.Task; +import org.gradle.api.file.RegularFileProperty; +import org.gradle.api.tasks.OutputFile; +import org.gradle.api.tasks.TaskAction; + +/** + * {@link Task} to create a {@link ResolvedBom resolved bom}. + * + * @author Andy Wilkinson + */ +public abstract class CreateResolvedBom extends DefaultTask { + + private final BomExtension bomExtension; + + private final BomResolver bomResolver; + + @Inject + public CreateResolvedBom(BomExtension bomExtension) { + getOutputs().upToDateWhen((spec) -> false); + this.bomExtension = bomExtension; + this.bomResolver = new BomResolver(getProject().getConfigurations(), getProject().getDependencies()); + getOutputFile().convention(getProject().getLayout().getBuildDirectory().file(getName() + "/resolved-bom.json")); + } + + @OutputFile + public abstract RegularFileProperty getOutputFile(); + + @TaskAction + void createResolvedBom() throws IOException { + ResolvedBom resolvedBom = this.bomResolver.resolve(this.bomExtension); + try (FileWriter writer = new FileWriter(getOutputFile().get().getAsFile())) { + resolvedBom.writeTo(writer); + } + } + +} diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/Library.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/Library.java index 6a99a434038f..7c77d60485fd 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/Library.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/Library.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,6 +37,7 @@ import org.gradle.api.artifacts.Configuration; import org.gradle.api.artifacts.Dependency; import org.gradle.api.artifacts.result.DependencyResult; +import org.gradle.api.artifacts.result.ResolutionResult; import org.springframework.boot.build.bom.bomr.version.DependencyVersion; @@ -304,9 +305,9 @@ public static class Group { private final List plugins; - private final List boms; + private final List boms; - public Group(String id, List modules, List plugins, List boms) { + public Group(String id, List modules, List plugins, List boms) { this.id = id; this.modules = modules; this.plugins = plugins; @@ -325,7 +326,7 @@ public List getPlugins() { return this.plugins; } - public List getBoms() { + public List getBoms() { return this.boms; } @@ -459,9 +460,8 @@ private Map resolveAligningDependencies() { Configuration alignmentConfiguration = this.project.getConfigurations() .detachedConfiguration(dependencies.toArray(new Dependency[0])); Map versions = new HashMap<>(); - for (DependencyResult dependency : alignmentConfiguration.getIncoming() - .getResolutionResult() - .getAllDependencies()) { + ResolutionResult resolutionResult = alignmentConfiguration.getIncoming().getResolutionResult(); + for (DependencyResult dependency : resolutionResult.getAllDependencies()) { versions.put(dependency.getFrom().getModuleVersion().getModule().toString(), dependency.getFrom().getModuleVersion().getVersion()); } @@ -516,7 +516,7 @@ private List getBomDependencies(Library manager) { .flatMap((group) -> group.getBoms() .stream() .map((bom) -> this.project.getDependencies() - .platform(group.getId() + ":" + bom + ":" + manager.getVersion().getVersion()))) + .platform(group.getId() + ":" + bom.name() + ":" + manager.getVersion().getVersion()))) .toList(); } @@ -571,4 +571,16 @@ public String url(LibraryVersion libraryVersion) { } + public record ImportedBom(String name, List permittedDependencies) { + + public ImportedBom(String name) { + this(name, Collections.emptyList()); + } + + } + + public record PermittedDependency(String groupId, String artifactId) { + + } + } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/ManagedDependencies.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/ManagedDependencies.java deleted file mode 100644 index cb6ca9581c3e..000000000000 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/ManagedDependencies.java +++ /dev/null @@ -1,127 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.build.bom; - -import java.io.File; -import java.io.FileReader; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.Set; - -import javax.xml.parsers.DocumentBuilderFactory; -import javax.xml.xpath.XPath; -import javax.xml.xpath.XPathConstants; -import javax.xml.xpath.XPathFactory; - -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import org.w3c.dom.NodeList; -import org.xml.sax.InputSource; - -import org.springframework.boot.build.bom.Library.Group; -import org.springframework.boot.build.bom.Library.Module; - -/** - * Managed dependencies from a bom or library. - * - * @author Andy Wilkinson - */ -class ManagedDependencies { - - private final Set ids; - - ManagedDependencies(Set ids) { - this.ids = ids; - } - - Set getIds() { - return this.ids; - } - - Difference diff(ManagedDependencies other) { - Set missing = new HashSet<>(this.ids); - missing.removeAll(other.ids); - Set unexpected = new HashSet<>(other.ids); - unexpected.removeAll(this.ids); - return new Difference(missing, unexpected); - } - - static ManagedDependencies ofBom(File bom) { - try { - Document bomDocument = DocumentBuilderFactory.newInstance() - .newDocumentBuilder() - .parse(new InputSource(new FileReader(bom))); - XPath xpath = XPathFactory.newInstance().newXPath(); - NodeList dependencyNodes = (NodeList) xpath - .evaluate("/project/dependencyManagement/dependencies/dependency", bomDocument, XPathConstants.NODESET); - NodeList propertyNodes = (NodeList) xpath.evaluate("/project/properties/*", bomDocument, - XPathConstants.NODESET); - Map properties = new HashMap<>(); - for (int i = 0; i < propertyNodes.getLength(); i++) { - Node property = propertyNodes.item(i); - String name = property.getNodeName(); - String value = property.getTextContent(); - properties.put("${%s}".formatted(name), value); - } - Set managedDependencies = new HashSet<>(); - for (int i = 0; i < dependencyNodes.getLength(); i++) { - Node dependency = dependencyNodes.item(i); - String groupId = (String) xpath.evaluate("groupId/text()", dependency, XPathConstants.STRING); - String artifactId = (String) xpath.evaluate("artifactId/text()", dependency, XPathConstants.STRING); - String version = (String) xpath.evaluate("version/text()", dependency, XPathConstants.STRING); - String classifier = (String) xpath.evaluate("classifier/text()", dependency, XPathConstants.STRING); - if (version.startsWith("${") && version.endsWith("}")) { - version = properties.get(version); - } - managedDependencies.add(asId(groupId, artifactId, version, classifier)); - } - return new ManagedDependencies(managedDependencies); - } - catch (Exception ex) { - throw new RuntimeException(ex); - } - } - - static String asId(String groupId, String artifactId, String version, String classifier) { - String id = groupId + ":" + artifactId + ":" + version; - if (classifier != null && !classifier.isEmpty()) { - id = id + ":" + classifier; - } - return id; - } - - static ManagedDependencies ofLibrary(Library library) { - Set managedByLibrary = new HashSet<>(); - for (Group group : library.getGroups()) { - for (Module module : group.getModules()) { - managedByLibrary.add(asId(group.getId(), module.getName(), library.getVersion().getVersion().toString(), - module.getClassifier())); - } - } - return new ManagedDependencies(managedByLibrary); - } - - record Difference(Set missing, Set unexpected) { - - boolean isEmpty() { - return this.missing.isEmpty() && this.unexpected.isEmpty(); - } - - } - -} diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/ResolvedBom.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/ResolvedBom.java new file mode 100644 index 000000000000..78fb43ecb634 --- /dev/null +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/ResolvedBom.java @@ -0,0 +1,121 @@ +/* + * Copyright 2025-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.build.bom; + +import java.io.File; +import java.io.FileReader; +import java.io.IOException; +import java.io.UncheckedIOException; +import java.io.Writer; +import java.net.URI; +import java.util.List; + +import com.fasterxml.jackson.annotation.JsonInclude.Include; +import com.fasterxml.jackson.annotation.JsonSetter; +import com.fasterxml.jackson.annotation.Nulls; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.SerializationFeature; + +/** + * A resolved bom. + * + * @author Andy Wilkinson + * @param id the ID of the resolved bom + * @param libraries the libraries declared in the bom + */ +public record ResolvedBom(Id id, List libraries) { + + private static final ObjectMapper objectMapper; + + static { + ObjectMapper mapper = new ObjectMapper().enable(SerializationFeature.INDENT_OUTPUT) + .setDefaultPropertyInclusion(Include.NON_EMPTY); + mapper.configOverride(List.class).setSetterInfo(JsonSetter.Value.forValueNulls(Nulls.AS_EMPTY)); + objectMapper = mapper; + } + + public static ResolvedBom readFrom(File file) { + try (FileReader reader = new FileReader(file)) { + return objectMapper.readValue(reader, ResolvedBom.class); + } + catch (IOException ex) { + throw new UncheckedIOException(ex); + } + } + + public void writeTo(Writer writer) { + try { + objectMapper.writeValue(writer, this); + } + catch (IOException ex) { + throw new UncheckedIOException(ex); + } + } + + public record ResolvedLibrary(String name, String version, String versionProperty, List managedDependencies, + List importedBoms, Links links) { + + } + + public record Id(String groupId, String artifactId, String version, String classifier) implements Comparable { + + Id(String groupId, String artifactId, String version) { + this(groupId, artifactId, version, null); + } + + @Override + public int compareTo(Id o) { + int result = this.groupId.compareTo(o.groupId); + if (result != 0) { + return result; + } + result = this.artifactId.compareTo(o.artifactId); + if (result != 0) { + return result; + } + return this.version.compareTo(o.version); + } + + @Override + public String toString() { + StringBuilder builder = new StringBuilder(); + builder.append(this.groupId); + builder.append(":"); + builder.append(this.artifactId); + builder.append(":"); + builder.append(this.version); + if (this.classifier != null) { + builder.append(this.classifier); + } + return builder.toString(); + } + + } + + public record Bom(Id id, Bom parent, List managedDependencies, List importedBoms) { + + } + + public record Links(List javadoc) { + + } + + public record JavadocLink(URI uri, List packages) { + + } + +} diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/InteractiveUpgradeResolver.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/InteractiveUpgradeResolver.java index 7b8b6e2492cc..b52328d70427 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/InteractiveUpgradeResolver.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/InteractiveUpgradeResolver.java @@ -74,7 +74,7 @@ private Upgrade resolveUpgrade(LibraryWithVersionOptions libraryWithVersionOptio } VersionOption defaultOption = defaultOption(library); VersionOption selected = selectOption(defaultOption, library, versionOptions); - return (selected.equals(defaultOption)) ? null : new Upgrade(library, selected.getVersion()); + return (selected.equals(defaultOption)) ? null : selected.upgrade(library); } private VersionOption defaultOption(Library library) { diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/MoveToSnapshots.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/MoveToSnapshots.java index a4147a74ee52..0cfa48cef7ce 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/MoveToSnapshots.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/MoveToSnapshots.java @@ -1,5 +1,5 @@ /* - * Copyright 2021-2024 the original author or authors. + * Copyright 2021-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,7 +19,7 @@ import java.time.OffsetDateTime; import java.util.List; import java.util.Map; -import java.util.function.BiPredicate; +import java.util.function.BiFunction; import javax.inject.Inject; @@ -77,27 +77,37 @@ protected boolean eligible(Library library) { } @Override - protected List> determineUpdatePredicates(Milestone milestone) { + protected BiFunction createVersionOptionResolver(Milestone milestone) { return switch (this.buildType) { - case OPEN_SOURCE -> determineOpenSourceUpdatePredicates(milestone); - case COMMERCIAL -> super.determineUpdatePredicates(milestone); + case OPEN_SOURCE -> createOpenSourceVersionOptionResolver(milestone); + case COMMERCIAL -> super.createVersionOptionResolver(milestone); }; } - private List> determineOpenSourceUpdatePredicates(Milestone milestone) { + private BiFunction createOpenSourceVersionOptionResolver( + Milestone milestone) { Map> scheduledReleases = getScheduledOpenSourceReleases(milestone); - List> predicates = super.determineUpdatePredicates(milestone); - predicates.add((library, candidate) -> { - List releases = scheduledReleases.get(library.getCalendarName()); - boolean match = (releases != null) - && releases.stream().anyMatch((release) -> candidate.isSnapshotFor(release.getVersion())); - if (logger.isInfoEnabled() && !match) { - logger.info("Ignoring {}. No release of {} scheduled before {}", candidate, library.getName(), - milestone.getDueOn()); + BiFunction resolver = super.createVersionOptionResolver(milestone); + return (library, dependencyVersion) -> { + VersionOption versionOption = resolver.apply(library, dependencyVersion); + if (versionOption != null) { + List releases = scheduledReleases.get(library.getCalendarName()); + if (releases != null) { + List matches = releases.stream() + .filter((release) -> dependencyVersion.isSnapshotFor(release.getVersion())) + .toList(); + if (!matches.isEmpty()) { + return new VersionOption.SnapshotVersionOption(versionOption.getVersion(), + matches.get(0).getVersion()); + } + } + if (logger.isInfoEnabled()) { + logger.info("Ignoring {}. No release of {} scheduled before {}", dependencyVersion, + library.getName(), milestone.getDueOn()); + } } - return match; - }); - return predicates; + return null; + }; } private Map> getScheduledOpenSourceReleases(Milestone milestone) { diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/StandardLibraryUpdateResolver.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/StandardLibraryUpdateResolver.java index adc16dd0c0da..474e044502f9 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/StandardLibraryUpdateResolver.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/StandardLibraryUpdateResolver.java @@ -24,13 +24,14 @@ import java.util.Map; import java.util.Set; import java.util.SortedSet; -import java.util.function.BiPredicate; +import java.util.function.BiFunction; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.boot.build.bom.Library; import org.springframework.boot.build.bom.Library.Group; +import org.springframework.boot.build.bom.Library.ImportedBom; import org.springframework.boot.build.bom.Library.Module; import org.springframework.boot.build.bom.Library.VersionAlignment; import org.springframework.boot.build.bom.bomr.version.DependencyVersion; @@ -46,13 +47,12 @@ class StandardLibraryUpdateResolver implements LibraryUpdateResolver { private final VersionResolver versionResolver; - private final BiPredicate predicate; + private final BiFunction versionOptionResolver; StandardLibraryUpdateResolver(VersionResolver versionResolver, - List> predicates) { + BiFunction versionOptionResolver) { this.versionResolver = versionResolver; - this.predicate = (library, dependencyVersion) -> predicates.stream() - .allMatch((predicate) -> predicate.test(library, dependencyVersion)); + this.versionOptionResolver = versionOptionResolver; } @Override @@ -112,22 +112,27 @@ private List determineResolvedVersionOptions(Library library) { moduleVersions.put(group.getId() + ":" + module.getName(), getLaterVersionsForModule(group.getId(), module.getName(), library)); } - for (String bom : group.getBoms()) { - moduleVersions.put(group.getId() + ":" + bom, getLaterVersionsForModule(group.getId(), bom, library)); + for (ImportedBom bom : group.getBoms()) { + moduleVersions.put(group.getId() + ":" + bom, + getLaterVersionsForModule(group.getId(), bom.name(), library)); } for (String plugin : group.getPlugins()) { moduleVersions.put(group.getId() + ":" + plugin, getLaterVersionsForModule(group.getId(), plugin, library)); } } - return moduleVersions.values() - .stream() - .flatMap(SortedSet::stream) - .distinct() - .filter((dependencyVersion) -> this.predicate.test(library, dependencyVersion)) - .map((version) -> (VersionOption) new VersionOption.ResolvedVersionOption(version, - getMissingModules(moduleVersions, version))) - .toList(); + List versionOptions = new ArrayList<>(); + moduleVersions.values().stream().flatMap(SortedSet::stream).distinct().forEach((dependencyVersion) -> { + VersionOption versionOption = this.versionOptionResolver.apply(library, dependencyVersion); + if (versionOption != null) { + List missingModules = getMissingModules(moduleVersions, dependencyVersion); + if (!missingModules.isEmpty()) { + versionOption = new VersionOption.ResolvedVersionOption(versionOption.getVersion(), missingModules); + } + versionOptions.add(versionOption); + } + }); + return versionOptions; } private List getMissingModules(Map> moduleVersions, diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/Upgrade.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/Upgrade.java index 4ce6a8cfbefd..647246b09eca 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/Upgrade.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/Upgrade.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,8 +17,6 @@ package org.springframework.boot.build.bom.bomr; import org.springframework.boot.build.bom.Library; -import org.springframework.boot.build.bom.Library.LibraryVersion; -import org.springframework.boot.build.bom.bomr.version.DependencyVersion; /** * An upgrade to change a {@link Library} to use a new version. @@ -31,19 +29,8 @@ */ record Upgrade(Library from, Library to, Library toRelease) { - Upgrade(Library from, DependencyVersion to) { - this(from, from.withVersion(new LibraryVersion(to))); - } - Upgrade(Library from, Library to) { - this(from, to, withReleaseVersion(to)); - } - - private static Library withReleaseVersion(Library to) { - String version = to.getVersion().toString(); - version = version.replace(".BUILD-SNAPSHOT", ""); - version = version.replace("-SNAPSHOT", ""); - return to.withVersion(new LibraryVersion(DependencyVersion.parse(version))); + this(from, to, to); } } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/UpgradeDependencies.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/UpgradeDependencies.java index 713a6b51e0de..2a3754a9524e 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/UpgradeDependencies.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/UpgradeDependencies.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,9 +24,11 @@ import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; +import java.util.Collections; import java.util.List; import java.util.Properties; import java.util.Set; +import java.util.function.BiFunction; import java.util.function.BiPredicate; import java.util.function.Predicate; import java.util.regex.Pattern; @@ -231,7 +233,7 @@ private List resolveUpgrades(Milestone milestone) { private LibraryUpdateResolver getLibraryUpdateResolver(Milestone milestone) { VersionResolver versionResolver = new MavenMetadataVersionResolver(getRepositories()); LibraryUpdateResolver libraryResolver = new StandardLibraryUpdateResolver(versionResolver, - determineUpdatePredicates(milestone)); + createVersionOptionResolver(milestone)); return new MultithreadedLibraryUpdateResolver(getThreads().get(), libraryResolver); } @@ -246,12 +248,19 @@ private List asRepositories(List repositoryName .toList(); } - protected List> determineUpdatePredicates(Milestone milestone) { + protected BiFunction createVersionOptionResolver(Milestone milestone) { List> updatePredicates = new ArrayList<>(); updatePredicates.add(this::compliesWithUpgradePolicy); updatePredicates.add(this::isAnUpgrade); updatePredicates.add(this::isNotProhibited); - return updatePredicates; + return (library, dependencyVersion) -> { + if (this.compliesWithUpgradePolicy(library, dependencyVersion) + && this.isAnUpgrade(library, dependencyVersion) + && this.isNotProhibited(library, dependencyVersion)) { + return new VersionOption.ResolvedVersionOption(dependencyVersion, Collections.emptyList()); + } + return null; + }; } private boolean compliesWithUpgradePolicy(Library library, DependencyVersion candidate) { diff --git a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/VersionOption.java b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/VersionOption.java index d1d33fd82389..d049ac04f91a 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/VersionOption.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/bom/bomr/VersionOption.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,6 +18,8 @@ import java.util.List; +import org.springframework.boot.build.bom.Library; +import org.springframework.boot.build.bom.Library.LibraryVersion; import org.springframework.boot.build.bom.Library.VersionAlignment; import org.springframework.boot.build.bom.bomr.version.DependencyVersion; import org.springframework.util.StringUtils; @@ -44,6 +46,10 @@ public String toString() { return this.version.toString(); } + Upgrade upgrade(Library library) { + return new Upgrade(library, library.withVersion(new LibraryVersion(this.version))); + } + static final class AlignedVersionOption extends VersionOption { private final VersionAlignment alignedWith; @@ -80,4 +86,26 @@ public String toString() { } + static final class SnapshotVersionOption extends VersionOption { + + private final DependencyVersion releaseVersion; + + SnapshotVersionOption(DependencyVersion version, DependencyVersion releaseVersion) { + super(version); + this.releaseVersion = releaseVersion; + } + + @Override + public String toString() { + return super.toString() + " (for " + this.releaseVersion + ")"; + } + + @Override + Upgrade upgrade(Library library) { + return new Upgrade(library, library.withVersion(new LibraryVersion(super.version)), + library.withVersion(new LibraryVersion(this.releaseVersion))); + } + + } + } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/constraints/DocumentConstrainedVersions.java b/buildSrc/src/main/java/org/springframework/boot/build/constraints/DocumentConstrainedVersions.java deleted file mode 100644 index e70502f785fc..000000000000 --- a/buildSrc/src/main/java/org/springframework/boot/build/constraints/DocumentConstrainedVersions.java +++ /dev/null @@ -1,63 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.build.constraints; - -import java.io.File; -import java.io.FileWriter; -import java.io.IOException; -import java.io.PrintWriter; - -import org.gradle.api.DefaultTask; -import org.gradle.api.file.RegularFileProperty; -import org.gradle.api.provider.SetProperty; -import org.gradle.api.tasks.Input; -import org.gradle.api.tasks.OutputFile; -import org.gradle.api.tasks.TaskAction; - -import org.springframework.boot.build.constraints.ExtractVersionConstraints.ConstrainedVersion; - -/** - * Task for documenting a platform's constrained versions. - * - * @author Andy Wilkinson - */ -public abstract class DocumentConstrainedVersions extends DefaultTask { - - @Input - public abstract SetProperty getConstrainedVersions(); - - @OutputFile - public abstract RegularFileProperty getOutputFile(); - - @TaskAction - public void documentConstrainedVersions() throws IOException { - File outputFile = getOutputFile().get().getAsFile(); - outputFile.getParentFile().mkdirs(); - try (PrintWriter writer = new PrintWriter(new FileWriter(outputFile))) { - writer.println("|==="); - writer.println("| Group ID | Artifact ID | Version"); - for (ConstrainedVersion constrainedVersion : getConstrainedVersions().get()) { - writer.println(); - writer.printf("| `%s`%n", constrainedVersion.getGroup()); - writer.printf("| `%s`%n", constrainedVersion.getArtifact()); - writer.printf("| `%s`%n", constrainedVersion.getVersion()); - } - writer.println("|==="); - } - } - -} diff --git a/buildSrc/src/main/java/org/springframework/boot/build/constraints/ExtractVersionConstraints.java b/buildSrc/src/main/java/org/springframework/boot/build/constraints/ExtractVersionConstraints.java deleted file mode 100644 index e1147c672d3e..000000000000 --- a/buildSrc/src/main/java/org/springframework/boot/build/constraints/ExtractVersionConstraints.java +++ /dev/null @@ -1,201 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.build.constraints; - -import java.io.Serializable; -import java.util.ArrayList; -import java.util.Collections; -import java.util.List; -import java.util.Map; -import java.util.Set; -import java.util.TreeMap; -import java.util.TreeSet; - -import org.gradle.api.DefaultTask; -import org.gradle.api.Project; -import org.gradle.api.Task; -import org.gradle.api.artifacts.ComponentMetadataDetails; -import org.gradle.api.artifacts.Configuration; -import org.gradle.api.artifacts.DependencyConstraint; -import org.gradle.api.artifacts.DependencyConstraintMetadata; -import org.gradle.api.artifacts.DependencyConstraintSet; -import org.gradle.api.artifacts.dsl.DependencyHandler; -import org.gradle.api.tasks.Internal; -import org.gradle.api.tasks.TaskAction; -import org.gradle.platform.base.Platform; - -import org.springframework.boot.build.bom.BomExtension; -import org.springframework.boot.build.bom.BomPlugin; -import org.springframework.boot.build.bom.Library; - -/** - * {@link Task} to extract constraints from a {@link Platform}. The platform's own - * constraints and those in any boms upon which it depends are extracted. - * - * @author Andy Wilkinson - */ -public abstract class ExtractVersionConstraints extends DefaultTask { - - private final Configuration configuration; - - private final Map versionConstraints = new TreeMap<>(); - - private final Set constrainedVersions = new TreeSet<>(); - - private final Set versionProperties = new TreeSet<>(); - - private final List dependencyConstraintSets = new ArrayList<>(); - - private final List boms = new ArrayList<>(); - - private final DependencyHandler dependencies; - - public ExtractVersionConstraints() { - this.dependencies = getProject().getDependencies(); - this.configuration = getProject().getConfigurations().create(getName()); - this.dependencies.getComponents().all(this::processMetadataDetails); - } - - public void enforcedPlatform(String projectPath) { - this.configuration.getDependencies() - .add(this.dependencies - .enforcedPlatform(this.dependencies.project(Collections.singletonMap("path", projectPath)))); - Project project = getProject().project(projectPath); - project.getPlugins().withType(BomPlugin.class).all((plugin) -> { - this.boms.add(project.getExtensions().getByType(BomExtension.class)); - this.dependencyConstraintSets - .add(project.getConfigurations().getByName("apiElements").getAllDependencyConstraints()); - }); - } - - @Internal - public Map getVersionConstraints() { - return Collections.unmodifiableMap(this.versionConstraints); - } - - @Internal - public Set getConstrainedVersions() { - return this.constrainedVersions; - } - - @Internal - public Set getVersionProperties() { - return this.versionProperties; - } - - @TaskAction - void extractVersionConstraints() { - this.configuration.resolve(); - this.boms.forEach(this::extractVersionProperties); - for (DependencyConstraintSet constraints : this.dependencyConstraintSets) { - for (DependencyConstraint constraint : constraints) { - this.versionConstraints.put(constraint.getGroup() + ":" + constraint.getName(), - constraint.getVersionConstraint().toString()); - this.constrainedVersions.add(new ConstrainedVersion(constraint.getGroup(), constraint.getName(), - constraint.getVersionConstraint().toString())); - } - } - } - - private void extractVersionProperties(BomExtension bomExtension) { - for (Library lib : bomExtension.getLibraries()) { - String versionProperty = lib.getVersionProperty(); - if (versionProperty != null) { - this.versionProperties.add(new VersionProperty(lib.getName(), versionProperty)); - } - } - } - - private void processMetadataDetails(ComponentMetadataDetails details) { - details.allVariants((variantMetadata) -> variantMetadata.withDependencyConstraints((dependencyConstraints) -> { - for (DependencyConstraintMetadata constraint : dependencyConstraints) { - this.versionConstraints.put(constraint.getGroup() + ":" + constraint.getName(), - constraint.getVersionConstraint().toString()); - this.constrainedVersions.add(new ConstrainedVersion(constraint.getGroup(), constraint.getName(), - constraint.getVersionConstraint().toString())); - } - })); - } - - public static final class ConstrainedVersion implements Comparable, Serializable { - - private final String group; - - private final String artifact; - - private final String version; - - private ConstrainedVersion(String group, String artifact, String version) { - this.group = group; - this.artifact = artifact; - this.version = version; - } - - public String getGroup() { - return this.group; - } - - public String getArtifact() { - return this.artifact; - } - - public String getVersion() { - return this.version; - } - - @Override - public int compareTo(ConstrainedVersion other) { - int groupComparison = this.group.compareTo(other.group); - if (groupComparison != 0) { - return groupComparison; - } - return this.artifact.compareTo(other.artifact); - } - - } - - public static final class VersionProperty implements Comparable, Serializable { - - private final String libraryName; - - private final String versionProperty; - - public VersionProperty(String libraryName, String versionProperty) { - this.libraryName = libraryName; - this.versionProperty = versionProperty; - } - - public String getLibraryName() { - return this.libraryName; - } - - public String getVersionProperty() { - return this.versionProperty; - } - - @Override - public int compareTo(VersionProperty other) { - int groupComparison = this.libraryName.compareToIgnoreCase(other.libraryName); - if (groupComparison != 0) { - return groupComparison; - } - return this.versionProperty.compareTo(other.versionProperty); - } - - } - -} diff --git a/buildSrc/src/main/java/org/springframework/boot/build/context/properties/DocumentConfigurationProperties.java b/buildSrc/src/main/java/org/springframework/boot/build/context/properties/DocumentConfigurationProperties.java index a7ac94f60ced..e54ac378151c 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/context/properties/DocumentConfigurationProperties.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/context/properties/DocumentConfigurationProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -87,7 +87,6 @@ private void corePrefixes(Config config) { config.accept("spring.autoconfigure"); config.accept("spring.banner"); config.accept("spring.beaninfo"); - config.accept("spring.codec"); config.accept("spring.config"); config.accept("spring.info"); config.accept("spring.jmx"); @@ -101,6 +100,7 @@ private void corePrefixes(Config config) { config.accept("spring.ssl"); config.accept("spring.task"); config.accept("spring.threads"); + config.accept("spring.validation"); config.accept("spring.mandatory-file-encoding"); config.accept("info"); config.accept("spring.output.ansi.enabled"); @@ -190,7 +190,6 @@ private void templatePrefixes(Config prefix) { prefix.accept("spring.groovy"); prefix.accept("spring.mustache"); prefix.accept("spring.thymeleaf"); - prefix.accept("spring.groovy.template.configuration", "See GroovyMarkupConfigurer"); } private void serverPrefixes(Config prefix) { diff --git a/buildSrc/src/main/java/org/springframework/boot/build/docs/ConfigureJavadocLinks.java b/buildSrc/src/main/java/org/springframework/boot/build/docs/ConfigureJavadocLinks.java new file mode 100644 index 000000000000..9f9bb13a3949 --- /dev/null +++ b/buildSrc/src/main/java/org/springframework/boot/build/docs/ConfigureJavadocLinks.java @@ -0,0 +1,72 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.build.docs; + +import java.net.URI; +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; + +import org.gradle.api.Action; +import org.gradle.api.file.FileCollection; +import org.gradle.api.tasks.javadoc.Javadoc; +import org.gradle.external.javadoc.StandardJavadocDocletOptions; + +import org.springframework.boot.build.bom.ResolvedBom; +import org.springframework.boot.build.bom.ResolvedBom.JavadocLink; + +/** + * An {@link Action} to configure the links option of a {@link Javadoc} task. + * + * @author Andy Wilkinson + */ +public class ConfigureJavadocLinks implements Action { + + private final FileCollection resolvedBoms; + + private final Collection includedLibraries; + + public ConfigureJavadocLinks(FileCollection resolvedBoms, Collection includedLibraries) { + this.resolvedBoms = resolvedBoms; + this.includedLibraries = includedLibraries; + } + + @Override + public void execute(Javadoc javadoc) { + javadoc.options((options) -> { + if (options instanceof StandardJavadocDocletOptions standardOptions) { + configureLinks(standardOptions); + } + }); + } + + private void configureLinks(StandardJavadocDocletOptions options) { + ResolvedBom resolvedBom = ResolvedBom.readFrom(this.resolvedBoms.getSingleFile()); + List links = new ArrayList<>(); + links.add("https://docs.oracle.com/en/java/javase/17/docs/api/"); + links.add("https://jakarta.ee/specifications/platform/9/apidocs/"); + resolvedBom.libraries() + .stream() + .filter((candidate) -> this.includedLibraries.contains(candidate.name())) + .flatMap((library) -> library.links().javadoc().stream()) + .map(JavadocLink::uri) + .map(URI::toString) + .forEach(links::add); + options.setLinks(links); + } + +} diff --git a/buildSrc/src/main/java/org/springframework/boot/build/docs/DocumentManagedDependencies.java b/buildSrc/src/main/java/org/springframework/boot/build/docs/DocumentManagedDependencies.java new file mode 100644 index 000000000000..74c42612f2d2 --- /dev/null +++ b/buildSrc/src/main/java/org/springframework/boot/build/docs/DocumentManagedDependencies.java @@ -0,0 +1,112 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.build.docs; + +import java.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.io.PrintWriter; +import java.util.Set; +import java.util.TreeSet; + +import org.gradle.api.DefaultTask; +import org.gradle.api.file.FileCollection; +import org.gradle.api.file.RegularFileProperty; +import org.gradle.api.tasks.InputFiles; +import org.gradle.api.tasks.OutputFile; +import org.gradle.api.tasks.PathSensitive; +import org.gradle.api.tasks.PathSensitivity; +import org.gradle.api.tasks.TaskAction; + +import org.springframework.boot.build.bom.ResolvedBom; +import org.springframework.boot.build.bom.ResolvedBom.Bom; +import org.springframework.boot.build.bom.ResolvedBom.Id; +import org.springframework.boot.build.bom.ResolvedBom.ResolvedLibrary; + +/** + * Task for documenting {@link ResolvedBom boms'} managed dependencies. + * + * @author Andy Wilkinson + */ +public abstract class DocumentManagedDependencies extends DefaultTask { + + private FileCollection resolvedBoms; + + @InputFiles + @PathSensitive(PathSensitivity.RELATIVE) + public FileCollection getResolvedBoms() { + return this.resolvedBoms; + } + + public void setResolvedBoms(FileCollection resolvedBoms) { + this.resolvedBoms = resolvedBoms; + } + + @OutputFile + public abstract RegularFileProperty getOutputFile(); + + @TaskAction + public void documentConstrainedVersions() throws IOException { + File outputFile = getOutputFile().get().getAsFile(); + outputFile.getParentFile().mkdirs(); + try (PrintWriter writer = new PrintWriter(new FileWriter(outputFile))) { + writer.println("|==="); + writer.println("| Group ID | Artifact ID | Version"); + Set managedCoordinates = new TreeSet<>((id1, id2) -> { + int result = id1.groupId().compareTo(id2.groupId()); + if (result != 0) { + return result; + } + return id1.artifactId().compareTo(id2.artifactId()); + }); + for (File file : getResolvedBoms().getFiles()) { + managedCoordinates.addAll(process(ResolvedBom.readFrom(file))); + } + for (Id id : managedCoordinates) { + writer.println(); + writer.printf("| `%s`%n", id.groupId()); + writer.printf("| `%s`%n", id.artifactId()); + writer.printf("| `%s`%n", id.version()); + } + writer.println("|==="); + } + } + + private Set process(ResolvedBom resolvedBom) { + TreeSet managedCoordinates = new TreeSet<>(); + for (ResolvedLibrary library : resolvedBom.libraries()) { + for (Id managedDependency : library.managedDependencies()) { + managedCoordinates.add(managedDependency); + } + for (Bom importedBom : library.importedBoms()) { + managedCoordinates.addAll(process(importedBom)); + } + } + return managedCoordinates; + } + + private Set process(Bom bom) { + TreeSet managedCoordinates = new TreeSet<>(); + bom.managedDependencies().stream().forEach(managedCoordinates::add); + Bom parent = bom.parent(); + if (parent != null) { + managedCoordinates.addAll(process(parent)); + } + return managedCoordinates; + } + +} diff --git a/buildSrc/src/main/java/org/springframework/boot/build/constraints/DocumentVersionProperties.java b/buildSrc/src/main/java/org/springframework/boot/build/docs/DocumentVersionProperties.java similarity index 54% rename from buildSrc/src/main/java/org/springframework/boot/build/constraints/DocumentVersionProperties.java rename to buildSrc/src/main/java/org/springframework/boot/build/docs/DocumentVersionProperties.java index ef1c7c8647ad..72bcf7716a6b 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/constraints/DocumentVersionProperties.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/docs/DocumentVersionProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -14,46 +14,66 @@ * limitations under the License. */ -package org.springframework.boot.build.constraints; +package org.springframework.boot.build.docs; import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.io.PrintWriter; +import java.util.List; import org.gradle.api.DefaultTask; +import org.gradle.api.file.FileCollection; import org.gradle.api.file.RegularFileProperty; -import org.gradle.api.provider.SetProperty; -import org.gradle.api.tasks.Input; +import org.gradle.api.tasks.InputFiles; import org.gradle.api.tasks.OutputFile; +import org.gradle.api.tasks.PathSensitive; +import org.gradle.api.tasks.PathSensitivity; import org.gradle.api.tasks.TaskAction; -import org.springframework.boot.build.constraints.ExtractVersionConstraints.VersionProperty; +import org.springframework.boot.build.bom.ResolvedBom; +import org.springframework.boot.build.bom.ResolvedBom.ResolvedLibrary; /** - * Task for documenting available version properties. + * Task for documenting {@link ResolvedBom boms'} version properties. * * @author Christoph Dreis + * @author Andy Wilkinson */ public abstract class DocumentVersionProperties extends DefaultTask { - @Input - public abstract SetProperty getVersionProperties(); + private FileCollection resolvedBoms; + + @InputFiles + @PathSensitive(PathSensitivity.RELATIVE) + public FileCollection getResolvedBoms() { + return this.resolvedBoms; + } + + public void setResolvedBoms(FileCollection resolvedBoms) { + this.resolvedBoms = resolvedBoms; + } @OutputFile public abstract RegularFileProperty getOutputFile(); @TaskAction public void documentVersionProperties() throws IOException { + List libraries = this.resolvedBoms.getFiles() + .stream() + .map(ResolvedBom::readFrom) + .flatMap((resolvedBom) -> resolvedBom.libraries().stream()) + .sorted((l1, l2) -> l1.name().compareToIgnoreCase(l2.name())) + .toList(); File outputFile = getOutputFile().getAsFile().get(); outputFile.getParentFile().mkdirs(); try (PrintWriter writer = new PrintWriter(new FileWriter(outputFile))) { writer.println("|==="); writer.println("| Library | Version Property"); - for (VersionProperty versionProperty : getVersionProperties().get()) { + for (ResolvedLibrary library : libraries) { writer.println(); - writer.printf("| `%s`%n", versionProperty.getLibraryName()); - writer.printf("| `%s`%n", versionProperty.getVersionProperty()); + writer.printf("| `%s`%n", library.name()); + writer.printf("| `%s`%n", library.versionProperty()); } writer.println("|==="); } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/mavenplugin/MavenPluginPlugin.java b/buildSrc/src/main/java/org/springframework/boot/build/mavenplugin/MavenPluginPlugin.java index ce576a1057da..6d61e6f07958 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/mavenplugin/MavenPluginPlugin.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/mavenplugin/MavenPluginPlugin.java @@ -17,10 +17,8 @@ package org.springframework.boot.build.mavenplugin; import java.io.File; -import java.io.FileInputStream; import java.io.FileWriter; import java.io.IOException; -import java.io.InputStream; import java.io.Writer; import java.nio.charset.StandardCharsets; import java.nio.file.Files; @@ -28,16 +26,9 @@ import java.nio.file.StandardCopyOption; import java.util.Arrays; import java.util.Properties; -import java.util.function.BiConsumer; +import java.util.Set; import javax.inject.Inject; -import javax.xml.parsers.DocumentBuilder; -import javax.xml.parsers.DocumentBuilderFactory; -import javax.xml.parsers.ParserConfigurationException; -import javax.xml.xpath.XPath; -import javax.xml.xpath.XPathConstants; -import javax.xml.xpath.XPathExpressionException; -import javax.xml.xpath.XPathFactory; import io.spring.javaformat.formatter.FileEdit; import io.spring.javaformat.formatter.FileFormatter; @@ -88,19 +79,18 @@ import org.gradle.api.tasks.bundling.Jar; import org.gradle.api.tasks.javadoc.Javadoc; import org.gradle.external.javadoc.StandardJavadocDocletOptions; -import org.w3c.dom.Document; import org.w3c.dom.Element; import org.w3c.dom.Node; import org.w3c.dom.NodeList; -import org.xml.sax.SAXException; import org.springframework.boot.build.DeployedPlugin; import org.springframework.boot.build.MavenRepositoryPlugin; +import org.springframework.boot.build.bom.ResolvedBom; +import org.springframework.boot.build.bom.ResolvedBom.ResolvedLibrary; import org.springframework.boot.build.optional.OptionalDependenciesPlugin; import org.springframework.boot.build.test.DockerTestPlugin; import org.springframework.boot.build.test.IntegrationTestPlugin; import org.springframework.core.CollectionFactory; -import org.springframework.util.Assert; /** * Plugin for building Spring Boot's Maven Plugin. @@ -125,7 +115,13 @@ public void apply(Project project) { generateHelpMojoTask); addDocumentPluginGoalsTask(project, generatePluginDescriptorTask); addPrepareMavenBinariesTask(project); - addExtractVersionPropertiesTask(project); + TaskProvider extractVersionPropertiesTask = addExtractVersionPropertiesTask(project); + project.getTasks() + .named(IntegrationTestPlugin.INT_TEST_TASK_NAME) + .configure((task) -> task.getInputs() + .file(extractVersionPropertiesTask.map(ExtractVersionProperties::getDestination)) + .withPathSensitivity(PathSensitivity.RELATIVE) + .withPropertyName("versionProperties")); publishOptionalDependenciesInPom(project); project.getTasks().withType(GenerateModuleMetadata.class).configureEach((task) -> task.setEnabled(false)); } @@ -335,9 +331,9 @@ private String replaceVersionPlaceholder(Project project, String input) { return input.replace("{{version}}", project.getVersion().toString()); } - private void addExtractVersionPropertiesTask(Project project) { - project.getTasks().register("extractVersionProperties", ExtractVersionProperties.class, (task) -> { - task.setEffectiveBoms(project.getConfigurations().create("versionProperties")); + private TaskProvider addExtractVersionPropertiesTask(Project project) { + return project.getTasks().register("extractVersionProperties", ExtractVersionProperties.class, (task) -> { + task.setResolvedBoms(project.getConfigurations().create("versionProperties")); task.getDestination() .set(project.getLayout() .getBuildDirectory() @@ -466,16 +462,16 @@ public void createRepository() { public abstract static class ExtractVersionProperties extends DefaultTask { - private FileCollection effectiveBoms; + private FileCollection resolvedBoms; @InputFiles @PathSensitive(PathSensitivity.RELATIVE) - public FileCollection getEffectiveBoms() { - return this.effectiveBoms; + public FileCollection getResolvedBoms() { + return this.resolvedBoms; } - public void setEffectiveBoms(FileCollection effectiveBoms) { - this.effectiveBoms = effectiveBoms; + public void setResolvedBoms(FileCollection resolvedBoms) { + this.resolvedBoms = resolvedBoms; } @OutputFile @@ -483,8 +479,8 @@ public void setEffectiveBoms(FileCollection effectiveBoms) { @TaskAction public void extractVersionProperties() { - EffectiveBom effectiveBom = new EffectiveBom(this.effectiveBoms.getSingleFile()); - Properties versions = extractVersionProperties(effectiveBom); + ResolvedBom resolvedBom = ResolvedBom.readFrom(this.resolvedBoms.getSingleFile()); + Properties versions = extractVersionProperties(resolvedBom); writeProperties(versions); } @@ -499,66 +495,18 @@ private void writeProperties(Properties versions) { } } - private Properties extractVersionProperties(EffectiveBom effectiveBom) { + private Properties extractVersionProperties(ResolvedBom resolvedBom) { Properties versions = CollectionFactory.createSortedProperties(true); - versions.setProperty("project.version", effectiveBom.version()); - effectiveBom.property("log4j2.version", versions::setProperty); - effectiveBom.property("maven-jar-plugin.version", versions::setProperty); - effectiveBom.property("maven-war-plugin.version", versions::setProperty); - effectiveBom.property("build-helper-maven-plugin.version", versions::setProperty); - effectiveBom.property("spring-framework.version", versions::setProperty); - effectiveBom.property("jakarta-servlet.version", versions::setProperty); - effectiveBom.property("kotlin.version", versions::setProperty); - effectiveBom.property("assertj.version", versions::setProperty); - effectiveBom.property("junit-jupiter.version", versions::setProperty); - return versions; - } - - } - - private static final class EffectiveBom { - - private final Document document; - - private final XPath xpath; - - private EffectiveBom(File bomFile) { - this.document = loadDocument(bomFile); - this.xpath = XPathFactory.newInstance().newXPath(); - } - - private Document loadDocument(File bomFile) { - try { - try (InputStream inputStream = new FileInputStream(bomFile)) { - DocumentBuilderFactory builderFactory = DocumentBuilderFactory.newInstance(); - DocumentBuilder builder = builderFactory.newDocumentBuilder(); - return builder.parse(inputStream); + versions.setProperty("project.version", resolvedBom.id().version()); + Set versionProperties = Set.of("log4j2.version", "maven-jar-plugin.version", + "maven-war-plugin.version", "build-helper-maven-plugin.version", "spring-framework.version", + "jakarta-servlet.version", "kotlin.version", "assertj.version", "junit-jupiter.version"); + for (ResolvedLibrary library : resolvedBom.libraries()) { + if (library.versionProperty() != null && versionProperties.contains(library.versionProperty())) { + versions.setProperty(library.versionProperty(), library.version()); } } - catch (ParserConfigurationException | SAXException | IOException ex) { - throw new IllegalStateException(ex); - } - } - - private String version() { - return get("version"); - } - - private void property(String name, BiConsumer handler) { - handler.accept(name, get("properties/" + name)); - } - - private String get(String expression) { - try { - Node node = (Node) this.xpath.compile("/project/" + expression) - .evaluate(this.document, XPathConstants.NODE); - String text = (node != null) ? node.getTextContent() : null; - Assert.hasLength(text, () -> "No result for expression " + expression); - return text; - } - catch (XPathExpressionException ex) { - throw new IllegalStateException(ex); - } + return versions; } } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/test/DockerTestBuildService.java b/buildSrc/src/main/java/org/springframework/boot/build/test/DockerTestBuildService.java index 2f70a735513c..016db7abe1ba 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/test/DockerTestBuildService.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/test/DockerTestBuildService.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,8 +22,10 @@ import org.gradle.api.services.BuildServiceParameters; /** - * Build service for Docker-based tests. Configured to only allow serial execution, - * thereby ensuring that Docker-based tests do not run in parallel. + * Build service for Docker-based tests. The maximum number of {@code dockerTest} tasks + * that can run in parallel can be configured using + * {@code org.springframework.boot.dockertest.max-parallel-tasks}. By default, only a + * single {@code dockerTest} task will run at a time. * * @author Andy Wilkinson */ @@ -32,7 +34,16 @@ abstract class DockerTestBuildService implements BuildService registerIfNecessary(Project project) { return project.getGradle() .getSharedServices() - .registerIfAbsent("dockerTest", DockerTestBuildService.class, (spec) -> spec.getMaxParallelUsages().set(1)); + .registerIfAbsent("dockerTest", DockerTestBuildService.class, + (spec) -> spec.getMaxParallelUsages().set(maxParallelTasks(project))); + } + + private static int maxParallelTasks(Project project) { + Object property = project.findProperty("org.springframework.boot.dockertest.max-parallel-tasks"); + if (property == null) { + return 1; + } + return Integer.parseInt(property.toString()); } } diff --git a/buildSrc/src/main/java/org/springframework/boot/build/test/SystemTestPlugin.java b/buildSrc/src/main/java/org/springframework/boot/build/test/SystemTestPlugin.java index 1fb2e5330cd6..b0ecb6cc98e6 100644 --- a/buildSrc/src/main/java/org/springframework/boot/build/test/SystemTestPlugin.java +++ b/buildSrc/src/main/java/org/springframework/boot/build/test/SystemTestPlugin.java @@ -64,6 +64,8 @@ private void configureSystemTesting(Project project) { .add(project.getConfigurations() .getByName(systemTestSourceSet.getRuntimeClasspathConfigurationName()))); }); + project.getDependencies() + .add(systemTestSourceSet.getRuntimeOnlyConfigurationName(), "org.junit.platform:junit-platform-launcher"); } private SourceSet createSourceSet(Project project) { diff --git a/buildSrc/src/main/resources/effective-bom-settings.xml b/buildSrc/src/main/resources/effective-bom-settings.xml deleted file mode 100644 index e2bb652e248e..000000000000 --- a/buildSrc/src/main/resources/effective-bom-settings.xml +++ /dev/null @@ -1,14 +0,0 @@ - - localRepositoryPath - - - spring-repositories - - true - - - - - - - diff --git a/buildSrc/src/test/java/org/springframework/boot/build/bom/BomPluginIntegrationTests.java b/buildSrc/src/test/java/org/springframework/boot/build/bom/BomPluginIntegrationTests.java index d6baf87b8893..ad86cc204e21 100644 --- a/buildSrc/src/test/java/org/springframework/boot/build/bom/BomPluginIntegrationTests.java +++ b/buildSrc/src/test/java/org/springframework/boot/build/bom/BomPluginIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -122,9 +122,7 @@ void libraryImportsAreIncludedInDependencyManagementOfGeneratedPom() throws Exce out.println("bom {"); out.println(" library('Jackson Bom', '2.10.0') {"); out.println(" group('com.fasterxml.jackson') {"); - out.println(" imports = ["); - out.println(" 'jackson-bom'"); - out.println(" ]"); + out.println(" bom('jackson-bom')"); out.println(" }"); out.println(" }"); out.println("}"); @@ -290,34 +288,6 @@ void libraryNamedSpringBootHasNoVersionProperty() throws IOException { }); } - // @Test - // void versionAlignmentIsVerified() throws IOException { - // try (PrintWriter out = new PrintWriter(new FileWriter(this.buildFile))) { - // out.println("plugins {"); - // out.println(" id 'org.springframework.boot.bom'"); - // out.println("}"); - // out.println("bom {"); - // out.println(" library('OAuth2 OIDC SDK', '8.36.1') {"); - // out.println(" alignedWith('Spring Security') {"); - // out.println( - // " - // source('https://github.com/spring-projects/spring-security/blob/${libraryVersion}/config/gradle/dependency-locks/optional.lockfile')"); - // out.println(" pattern('com.nimbusds:oauth2-oidc-sdk:(.+)')"); - // out.println(" }"); - // out.println(" group('com.nimbusds') {"); - // out.println(" modules = ["); - // out.println(" 'oauth2-oidc-sdk'"); - // out.println(" ]"); - // out.println(" }"); - // out.println(" }"); - // out.println(" library('Spring Security', '5.4.7') {"); - // out.println(" }"); - // out.println("}"); - // } - // System.out.println(runGradle(DeployedPlugin.GENERATE_POM_TASK_NAME, - // "-s").getOutput()); - // } - private BuildResult runGradle(String... args) { return GradleRunner.create() .withDebug(true) diff --git a/buildSrc/src/test/java/org/springframework/boot/build/bom/bomr/UpgradeApplicatorTests.java b/buildSrc/src/test/java/org/springframework/boot/build/bom/bomr/UpgradeApplicatorTests.java index b83742ad3691..fca394cae200 100644 --- a/buildSrc/src/test/java/org/springframework/boot/build/bom/bomr/UpgradeApplicatorTests.java +++ b/buildSrc/src/test/java/org/springframework/boot/build/bom/bomr/UpgradeApplicatorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -52,11 +52,10 @@ void whenUpgradeIsAppliedToLibraryWithVersionThenBomIsUpdated() throws IOExcepti String originalContents = Files.readString(bom.toPath()); File gradleProperties = new File(this.temp, "gradle.properties"); FileCopyUtils.copy(new File("src/test/resources/gradle.properties"), gradleProperties); - new UpgradeApplicator(bom.toPath(), gradleProperties.toPath()).apply( - new Upgrade( - new Library("ActiveMQ", null, new LibraryVersion(DependencyVersion.parse("5.15.11")), null, - null, false, null, null, null, Collections.emptyMap()), - DependencyVersion.parse("5.16"))); + Library activeMq = new Library("ActiveMQ", null, new LibraryVersion(DependencyVersion.parse("5.15.11")), null, + null, false, null, null, null, Collections.emptyMap()); + new UpgradeApplicator(bom.toPath(), gradleProperties.toPath()) + .apply(new Upgrade(activeMq, activeMq.withVersion(new LibraryVersion(DependencyVersion.parse("5.16"))))); String bomContents = Files.readString(bom.toPath()); assertThat(bomContents).hasSize(originalContents.length() - 3); } @@ -67,9 +66,10 @@ void whenUpgradeIsAppliedToLibraryWithVersionPropertyThenGradlePropertiesIsUpdat FileCopyUtils.copy(new File("src/test/resources/bom.gradle"), bom); File gradleProperties = new File(this.temp, "gradle.properties"); FileCopyUtils.copy(new File("src/test/resources/gradle.properties"), gradleProperties); + Library kotlin = new Library("Kotlin", null, new LibraryVersion(DependencyVersion.parse("1.3.70")), null, null, + false, null, null, null, Collections.emptyMap()); new UpgradeApplicator(bom.toPath(), gradleProperties.toPath()) - .apply(new Upgrade(new Library("Kotlin", null, new LibraryVersion(DependencyVersion.parse("1.3.70")), null, - null, false, null, null, null, Collections.emptyMap()), DependencyVersion.parse("1.4"))); + .apply(new Upgrade(kotlin, kotlin.withVersion(new LibraryVersion(DependencyVersion.parse("1.4"))))); Properties properties = new Properties(); try (InputStream in = new FileInputStream(gradleProperties)) { properties.load(in); diff --git a/buildSrc/src/test/java/org/springframework/boot/build/bom/bomr/UpgradeTests.java b/buildSrc/src/test/java/org/springframework/boot/build/bom/bomr/UpgradeTests.java index ee7a0c3e9b47..245589a9c593 100644 --- a/buildSrc/src/test/java/org/springframework/boot/build/bom/bomr/UpgradeTests.java +++ b/buildSrc/src/test/java/org/springframework/boot/build/bom/bomr/UpgradeTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2024-2024 the original author or authors. + * Copyright 2024-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ class UpgradeTests { void createToRelease() { Library from = new Library("Test", null, new LibraryVersion(DependencyVersion.parse("1.0.0")), null, null, false, null, null, null, null); - Upgrade upgrade = new Upgrade(from, DependencyVersion.parse("1.0.1")); + Upgrade upgrade = new Upgrade(from, from.withVersion(new LibraryVersion(DependencyVersion.parse("1.0.1")))); assertThat(upgrade.from().getNameAndVersion()).isEqualTo("Test 1.0.0"); assertThat(upgrade.to().getNameAndVersion()).isEqualTo("Test 1.0.1"); assertThat(upgrade.toRelease().getNameAndVersion()).isEqualTo("Test 1.0.1"); @@ -45,7 +45,9 @@ void createToRelease() { void createToSnapshot() { Library from = new Library("Test", null, new LibraryVersion(DependencyVersion.parse("1.0.0")), null, null, false, null, null, null, null); - Upgrade upgrade = new Upgrade(from, DependencyVersion.parse("1.0.1-SNAPSHOT")); + Upgrade upgrade = new Upgrade(from, + from.withVersion(new LibraryVersion(DependencyVersion.parse("1.0.1-SNAPSHOT"))), + from.withVersion(new LibraryVersion(DependencyVersion.parse("1.0.1")))); assertThat(upgrade.from().getNameAndVersion()).isEqualTo("Test 1.0.0"); assertThat(upgrade.to().getNameAndVersion()).isEqualTo("Test 1.0.1-SNAPSHOT"); assertThat(upgrade.toRelease().getNameAndVersion()).isEqualTo("Test 1.0.1"); diff --git a/eclipse/spring-boot-project.setup b/eclipse/spring-boot-project.setup index 5e3eef6fc1ae..868627550cfe 100644 --- a/eclipse/spring-boot-project.setup +++ b/eclipse/spring-boot-project.setup @@ -11,8 +11,8 @@ xmlns:setup.workingsets="http://www.eclipse.org/oomph/setup/workingsets/1.0" xmlns:workingsets="http://www.eclipse.org/oomph/workingsets/1.0" xsi:schemaLocation="http://www.eclipse.org/oomph/setup/jdt/1.0 http://git.eclipse.org/c/oomph/org.eclipse.oomph.git/plain/setups/models/JDT.ecore http://www.eclipse.org/buildship/oomph/1.0 https://raw.githubusercontent.com/eclipse/buildship/master/org.eclipse.buildship.oomph/model/GradleImport-1.0.ecore http://www.eclipse.org/oomph/predicates/1.0 http://git.eclipse.org/c/oomph/org.eclipse.oomph.git/plain/setups/models/Predicates.ecore http://www.eclipse.org/oomph/setup/workingsets/1.0 http://git.eclipse.org/c/oomph/org.eclipse.oomph.git/plain/setups/models/SetupWorkingSets.ecore http://www.eclipse.org/oomph/workingsets/1.0 http://git.eclipse.org/c/oomph/org.eclipse.oomph.git/plain/setups/models/WorkingSets.ecore" - name="spring.boot.3.4.x" - label="Spring Boot 3.4.x"> + name="spring.boot.3.5.x" + label="Spring Boot 3.5.x"> names = new LinkedHashSet<>(groups.getNames()); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/availability/DelegatingAvailabilityProbesHealthEndpointGroup.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/availability/DelegatingAvailabilityProbesHealthEndpointGroup.java index 66975852e7cd..daae8f4edf07 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/availability/DelegatingAvailabilityProbesHealthEndpointGroup.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/availability/DelegatingAvailabilityProbesHealthEndpointGroup.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ class DelegatingAvailabilityProbesHealthEndpointGroup implements HealthEndpointG DelegatingAvailabilityProbesHealthEndpointGroup(HealthEndpointGroup delegate, AdditionalHealthEndpointPath additionalPath) { - Assert.notNull(delegate, "Delegate must not be null"); + Assert.notNull(delegate, "'delegate' must not be null"); this.delegate = delegate; this.additionalPath = additionalPath; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/CloudFoundryWebEndpointDiscoverer.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/CloudFoundryWebEndpointDiscoverer.java index 27763edec389..b703e993de09 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/CloudFoundryWebEndpointDiscoverer.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/CloudFoundryWebEndpointDiscoverer.java @@ -57,7 +57,7 @@ public class CloudFoundryWebEndpointDiscoverer extends WebEndpointDiscoverer { * @param endpointPathMappers the endpoint path mappers * @param invokerAdvisors invoker advisors to apply * @param endpointFilters endpoint filters to apply - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link #CloudFoundryWebEndpointDiscoverer(ApplicationContext, ParameterValueMapper, EndpointMediaTypes, List, Collection, Collection, Collection)} */ @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundryActuatorAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundryActuatorAutoConfiguration.java index f19dd0d744b5..4b0dde25caf9 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundryActuatorAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundryActuatorAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -46,10 +46,10 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnCloudPlatform; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.cloud.CloudPlatform; import org.springframework.boot.info.GitProperties; @@ -76,7 +76,7 @@ * @since 2.0.0 */ @AutoConfiguration(after = { HealthEndpointAutoConfiguration.class, InfoEndpointAutoConfiguration.class }) -@ConditionalOnProperty(prefix = "management.cloudfoundry", name = "enabled", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "management.cloudfoundry.enabled", matchIfMissing = true) @ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.REACTIVE) @ConditionalOnCloudPlatform(CloudPlatform.CLOUD_FOUNDRY) public class ReactiveCloudFoundryActuatorAutoConfiguration { diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundrySecurityService.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundrySecurityService.java index 18ac9811b3df..e42d2e41428b 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundrySecurityService.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundrySecurityService.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,8 +56,8 @@ class ReactiveCloudFoundrySecurityService { ReactiveCloudFoundrySecurityService(WebClient.Builder webClientBuilder, String cloudControllerUrl, boolean skipSslValidation) { - Assert.notNull(webClientBuilder, "WebClient must not be null"); - Assert.notNull(cloudControllerUrl, "CloudControllerUrl must not be null"); + Assert.notNull(webClientBuilder, "'webClientBuilder' must not be null"); + Assert.notNull(cloudControllerUrl, "'cloudControllerUrl' must not be null"); if (skipSslValidation) { webClientBuilder.clientConnector(buildTrustAllSslConnector()); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundryActuatorAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundryActuatorAutoConfiguration.java index f4c234712dae..1e40af4cd48e 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundryActuatorAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundryActuatorAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,10 +43,10 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnCloudPlatform; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.security.SecurityProperties; import org.springframework.boot.cloud.CloudPlatform; @@ -77,7 +77,7 @@ */ @AutoConfiguration(after = { ServletManagementContextAutoConfiguration.class, HealthEndpointAutoConfiguration.class, InfoEndpointAutoConfiguration.class }) -@ConditionalOnProperty(prefix = "management.cloudfoundry", name = "enabled", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "management.cloudfoundry.enabled", matchIfMissing = true) @ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.SERVLET) @ConditionalOnClass(DispatcherServlet.class) @ConditionalOnBean(DispatcherServlet.class) diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundrySecurityService.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundrySecurityService.java index 9be59775512b..4a839437592f 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundrySecurityService.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundrySecurityService.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,8 +49,8 @@ class CloudFoundrySecurityService { CloudFoundrySecurityService(RestTemplateBuilder restTemplateBuilder, String cloudControllerUrl, boolean skipSslValidation) { - Assert.notNull(restTemplateBuilder, "RestTemplateBuilder must not be null"); - Assert.notNull(cloudControllerUrl, "CloudControllerUrl must not be null"); + Assert.notNull(restTemplateBuilder, "'restTemplateBuilder' must not be null"); + Assert.notNull(cloudControllerUrl, "'cloudControllerUrl' must not be null"); if (skipSslValidation) { restTemplateBuilder = restTemplateBuilder.requestFactory(SkipSslVerificationHttpRequestFactory.class); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/EndpointExposure.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/EndpointExposure.java index 967f72bdf637..5d5bd55d41a4 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/EndpointExposure.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/EndpointExposure.java @@ -37,7 +37,7 @@ public enum EndpointExposure { /** * Exposed on Cloud Foundry over `/cloudfoundryapplication`. * @since 2.6.4 - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of using + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of using * {@link EndpointExposure#WEB} */ @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/IncludeExcludeEndpointFilter.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/IncludeExcludeEndpointFilter.java index 00affa4cfff3..07b3dc1840ef 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/IncludeExcludeEndpointFilter.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/IncludeExcludeEndpointFilter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -80,10 +80,10 @@ public IncludeExcludeEndpointFilter(Class endpointType, Collection in private IncludeExcludeEndpointFilter(Class endpointType, Environment environment, String prefix, EndpointPatterns defaultIncludes) { - Assert.notNull(endpointType, "EndpointType must not be null"); - Assert.notNull(environment, "Environment must not be null"); - Assert.hasText(prefix, "Prefix must not be empty"); - Assert.notNull(defaultIncludes, "DefaultIncludes must not be null"); + Assert.notNull(endpointType, "'endpointType' must not be null"); + Assert.notNull(environment, "'environment' must not be null"); + Assert.hasText(prefix, "'prefix' must not be empty"); + Assert.notNull(defaultIncludes, "'defaultIncludes' must not be null"); Binder binder = Binder.get(environment); this.endpointType = endpointType; this.include = new EndpointPatterns(bind(binder, prefix + ".include")); @@ -93,8 +93,8 @@ private IncludeExcludeEndpointFilter(Class endpointType, Environment environm private IncludeExcludeEndpointFilter(Class endpointType, Collection include, Collection exclude, EndpointPatterns defaultIncludes) { - Assert.notNull(endpointType, "EndpointType Type must not be null"); - Assert.notNull(defaultIncludes, "DefaultIncludes must not be null"); + Assert.notNull(endpointType, "'endpointType' Type must not be null"); + Assert.notNull(defaultIncludes, "'defaultIncludes' must not be null"); this.endpointType = endpointType; this.include = new EndpointPatterns(include); this.defaultIncludes = defaultIncludes; diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/jackson/JacksonEndpointAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/jackson/JacksonEndpointAutoConfiguration.java index 9a409fa55d63..2add75776fbf 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/jackson/JacksonEndpointAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/jackson/JacksonEndpointAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,8 +23,8 @@ import org.springframework.boot.actuate.endpoint.jackson.EndpointObjectMapper; import org.springframework.boot.autoconfigure.AutoConfigureAfter; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.jackson.JacksonAutoConfiguration; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -41,7 +41,7 @@ public class JacksonEndpointAutoConfiguration { @Bean - @ConditionalOnProperty(name = "management.endpoints.jackson.isolated-object-mapper", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "management.endpoints.jackson.isolated-object-mapper", matchIfMissing = true) @ConditionalOnClass({ ObjectMapper.class, Jackson2ObjectMapperBuilder.class }) public EndpointObjectMapper endpointObjectMapper() { ObjectMapper objectMapper = Jackson2ObjectMapperBuilder.json() diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/jmx/JmxEndpointAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/jmx/JmxEndpointAutoConfiguration.java index 6273598c5d08..da4342959ca6 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/jmx/JmxEndpointAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/jmx/JmxEndpointAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,8 +41,8 @@ import org.springframework.boot.actuate.endpoint.jmx.annotation.JmxEndpointDiscoverer; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnSingleCandidate; import org.springframework.boot.autoconfigure.condition.SearchStrategy; import org.springframework.boot.autoconfigure.jmx.JmxAutoConfiguration; @@ -63,7 +63,7 @@ */ @AutoConfiguration(after = { JmxAutoConfiguration.class, EndpointAutoConfiguration.class }) @EnableConfigurationProperties({ JmxEndpointProperties.class, JmxProperties.class }) -@ConditionalOnProperty(prefix = "spring.jmx", name = "enabled", havingValue = "true") +@ConditionalOnBooleanProperty("spring.jmx.enabled") public class JmxEndpointAutoConfiguration { private final ApplicationContext applicationContext; diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/CorsEndpointProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/CorsEndpointProperties.java index 287b70f921f9..35f73f4bb7f9 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/CorsEndpointProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/CorsEndpointProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,7 +33,7 @@ * @author Andy Wilkinson * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.endpoints.web.cors") +@ConfigurationProperties("management.endpoints.web.cors") public class CorsEndpointProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/WebEndpointProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/WebEndpointProperties.java index 06a9223eaf62..2982e7958d7f 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/WebEndpointProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/WebEndpointProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ * @author Phillip Webb * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.endpoints.web") +@ConfigurationProperties("management.endpoints.web") public class WebEndpointProperties { private final Exposure exposure = new Exposure(); @@ -62,7 +62,7 @@ public String getBasePath() { } public void setBasePath(String basePath) { - Assert.isTrue(basePath.isEmpty() || basePath.startsWith("/"), "Base path must start with '/' or be empty"); + Assert.isTrue(basePath.isEmpty() || basePath.startsWith("/"), "'basePath' must start with '/' or be empty"); this.basePath = cleanBasePath(basePath); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/health/AbstractCompositeHealthContributorConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/health/AbstractCompositeHealthContributorConfiguration.java index 73e414e82115..b0966d9dffe9 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/health/AbstractCompositeHealthContributorConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/health/AbstractCompositeHealthContributorConfiguration.java @@ -61,7 +61,7 @@ protected final C createContributor(ConfigurableListableBeanFactory beanFactory, } protected final C createContributor(Map beans) { - Assert.notEmpty(beans, "Beans must not be empty"); + Assert.notEmpty(beans, "'beans' must not be empty"); if (beans.size() == 1) { return createIndicator(beans.values().iterator().next()); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/health/HealthEndpointConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/health/HealthEndpointConfiguration.java index c4b78bb3b9c7..7006182f0231 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/health/HealthEndpointConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/health/HealthEndpointConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,8 +43,8 @@ import org.springframework.boot.actuate.health.SimpleHttpCodeStatusMapper; import org.springframework.boot.actuate.health.SimpleStatusAggregator; import org.springframework.boot.actuate.health.StatusAggregator; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.ApplicationContext; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -91,8 +91,7 @@ HealthContributorRegistry healthContributorRegistry(ApplicationContext applicati } @Bean - @ConditionalOnProperty(name = "management.endpoint.health.validate-group-membership", havingValue = "true", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "management.endpoint.health.validate-group-membership", matchIfMissing = true) HealthEndpointGroupMembershipValidator healthEndpointGroupMembershipValidator(HealthEndpointProperties properties, HealthContributorRegistry healthContributorRegistry) { return new HealthEndpointGroupMembershipValidator(properties, healthContributorRegistry); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthContributorAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthContributorAutoConfiguration.java index 379eb09e36de..a8584bd50141 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthContributorAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthContributorAutoConfiguration.java @@ -101,7 +101,7 @@ public HealthContributor dbHealthContributor(ConfigurableListableBeanFactory bea } private HealthContributor createContributor(Map beans) { - Assert.notEmpty(beans, "Beans must not be empty"); + Assert.notEmpty(beans, "'beans' must not be empty"); if (beans.size() == 1) { return createContributor(beans.values().iterator().next()); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthIndicatorProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthIndicatorProperties.java index 5efe2583e7f0..5adf77f4e2a5 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthIndicatorProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthIndicatorProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ * @author Julio Gomez * @since 2.4.0 */ -@ConfigurationProperties(prefix = "management.health.db") +@ConfigurationProperties("management.health.db") public class DataSourceHealthIndicatorProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/logging/LogFileWebEndpointProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/logging/LogFileWebEndpointProperties.java index a13b0858f998..3d2dfb60e910 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/logging/LogFileWebEndpointProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/logging/LogFileWebEndpointProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.endpoint.logfile") +@ConfigurationProperties("management.endpoint.logfile") public class LogFileWebEndpointProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/logging/otlp/OtlpLoggingConfigurations.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/logging/otlp/OtlpLoggingConfigurations.java index 14b491bd237d..d3c93f771617 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/logging/otlp/OtlpLoggingConfigurations.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/logging/otlp/OtlpLoggingConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,11 +18,13 @@ import java.util.Locale; +import io.opentelemetry.api.metrics.MeterProvider; import io.opentelemetry.exporter.otlp.http.logs.OtlpHttpLogRecordExporter; import io.opentelemetry.exporter.otlp.http.logs.OtlpHttpLogRecordExporterBuilder; import io.opentelemetry.exporter.otlp.logs.OtlpGrpcLogRecordExporter; import io.opentelemetry.exporter.otlp.logs.OtlpGrpcLogRecordExporterBuilder; +import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.actuate.autoconfigure.logging.ConditionalOnEnabledLoggingExport; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; @@ -46,7 +48,7 @@ static class ConnectionDetails { @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(prefix = "management.otlp.logging", name = "endpoint") + @ConditionalOnProperty("management.otlp.logging.endpoint") OtlpLoggingConnectionDetails otlpLoggingConnectionDetails(OtlpLoggingProperties properties) { return new PropertiesOtlpLoggingConnectionDetails(properties); } @@ -81,29 +83,30 @@ public String getUrl(Transport transport) { static class Exporters { @Bean - @ConditionalOnProperty(prefix = "management.otlp.logging", name = "transport", havingValue = "http", - matchIfMissing = true) + @ConditionalOnProperty(name = "management.otlp.logging.transport", havingValue = "http", matchIfMissing = true) OtlpHttpLogRecordExporter otlpHttpLogRecordExporter(OtlpLoggingProperties properties, - OtlpLoggingConnectionDetails connectionDetails) { + OtlpLoggingConnectionDetails connectionDetails, ObjectProvider meterProvider) { OtlpHttpLogRecordExporterBuilder builder = OtlpHttpLogRecordExporter.builder() .setEndpoint(connectionDetails.getUrl(Transport.HTTP)) .setTimeout(properties.getTimeout()) .setConnectTimeout(properties.getConnectTimeout()) .setCompression(properties.getCompression().name().toLowerCase(Locale.US)); properties.getHeaders().forEach(builder::addHeader); + meterProvider.ifAvailable(builder::setMeterProvider); return builder.build(); } @Bean - @ConditionalOnProperty(prefix = "management.otlp.logging", name = "transport", havingValue = "grpc") + @ConditionalOnProperty(name = "management.otlp.logging.transport", havingValue = "grpc") OtlpGrpcLogRecordExporter otlpGrpcLogRecordExporter(OtlpLoggingProperties properties, - OtlpLoggingConnectionDetails connectionDetails) { + OtlpLoggingConnectionDetails connectionDetails, ObjectProvider meterProvider) { OtlpGrpcLogRecordExporterBuilder builder = OtlpGrpcLogRecordExporter.builder() .setEndpoint(connectionDetails.getUrl(Transport.GRPC)) .setTimeout(properties.getTimeout()) .setConnectTimeout(properties.getConnectTimeout()) .setCompression(properties.getCompression().name().toLowerCase(Locale.US)); properties.getHeaders().forEach(builder::addHeader); + meterProvider.ifAvailable(builder::setMeterProvider); return builder.build(); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/JvmMetricsAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/JvmMetricsAutoConfiguration.java index 1ac29acea7d4..dada651f0986 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/JvmMetricsAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/JvmMetricsAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,6 +17,7 @@ package org.springframework.boot.actuate.autoconfigure.metrics; import io.micrometer.core.instrument.MeterRegistry; +import io.micrometer.core.instrument.binder.MeterBinder; import io.micrometer.core.instrument.binder.jvm.ClassLoaderMetrics; import io.micrometer.core.instrument.binder.jvm.JvmCompilationMetrics; import io.micrometer.core.instrument.binder.jvm.JvmGcMetrics; @@ -25,12 +26,18 @@ import io.micrometer.core.instrument.binder.jvm.JvmMemoryMetrics; import io.micrometer.core.instrument.binder.jvm.JvmThreadMetrics; +import org.springframework.aot.hint.MemberCategory; +import org.springframework.aot.hint.RuntimeHints; +import org.springframework.aot.hint.RuntimeHintsRegistrar; +import org.springframework.beans.BeanUtils; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.ImportRuntimeHints; +import org.springframework.util.ClassUtils; /** * {@link EnableAutoConfiguration Auto-configuration} for JVM metrics. @@ -44,6 +51,8 @@ @ConditionalOnBean(MeterRegistry.class) public class JvmMetricsAutoConfiguration { + private static final String VIRTUAL_THREAD_METRICS_CLASS = "io.micrometer.java21.instrument.binder.jdk.VirtualThreadMetrics"; + @Bean @ConditionalOnMissingBean public JvmGcMetrics jvmGcMetrics() { @@ -86,4 +95,25 @@ public JvmCompilationMetrics jvmCompilationMetrics() { return new JvmCompilationMetrics(); } + @Bean + @ConditionalOnClass(name = VIRTUAL_THREAD_METRICS_CLASS) + @ConditionalOnMissingBean(type = VIRTUAL_THREAD_METRICS_CLASS) + @ImportRuntimeHints(VirtualThreadMetricsRuntimeHintsRegistrar.class) + MeterBinder virtualThreadMetrics() throws ClassNotFoundException { + Class virtualThreadMetricsClass = ClassUtils.forName(VIRTUAL_THREAD_METRICS_CLASS, + getClass().getClassLoader()); + return (MeterBinder) BeanUtils.instantiateClass(virtualThreadMetricsClass); + } + + static final class VirtualThreadMetricsRuntimeHintsRegistrar implements RuntimeHintsRegistrar { + + @Override + public void registerHints(RuntimeHints hints, ClassLoader classLoader) { + hints.reflection() + .registerTypeIfPresent(classLoader, VIRTUAL_THREAD_METRICS_CLASS, + MemberCategory.INVOKE_PUBLIC_CONSTRUCTORS); + } + + } + } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/MetricsAspectsAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/MetricsAspectsAutoConfiguration.java index 959789adfd50..0c680cd7e234 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/MetricsAspectsAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/MetricsAspectsAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,16 +23,13 @@ import org.aspectj.weaver.Advice; import org.springframework.beans.factory.ObjectProvider; -import org.springframework.boot.actuate.autoconfigure.metrics.MetricsAspectsAutoConfiguration.ObservationAnnotationsEnabledCondition; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; -import org.springframework.boot.autoconfigure.condition.AnyNestedCondition; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Conditional; /** * {@link EnableAutoConfiguration Auto-configuration} for Micrometer-based metrics @@ -43,7 +40,7 @@ */ @AutoConfiguration(after = { MetricsAutoConfiguration.class, CompositeMeterRegistryAutoConfiguration.class }) @ConditionalOnClass({ MeterRegistry.class, Advice.class }) -@Conditional(ObservationAnnotationsEnabledCondition.class) +@ConditionalOnBooleanProperty("management.observations.annotations.enabled") @ConditionalOnBean(MeterRegistry.class) public class MetricsAspectsAutoConfiguration { @@ -62,22 +59,4 @@ TimedAspect timedAspect(MeterRegistry registry, return timedAspect; } - static final class ObservationAnnotationsEnabledCondition extends AnyNestedCondition { - - ObservationAnnotationsEnabledCondition() { - super(ConfigurationPhase.PARSE_CONFIGURATION); - } - - @ConditionalOnProperty(prefix = "micrometer.observations.annotations", name = "enabled", havingValue = "true") - static class MicrometerObservationsEnabledCondition { - - } - - @ConditionalOnProperty(prefix = "management.observations.annotations", name = "enabled", havingValue = "true") - static class ManagementObservationsEnabledCondition { - - } - - } - } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/OnlyOnceLoggingDenyMeterFilter.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/OnlyOnceLoggingDenyMeterFilter.java index b257f66c4fb4..51ba32514837 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/OnlyOnceLoggingDenyMeterFilter.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/OnlyOnceLoggingDenyMeterFilter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,7 +43,7 @@ public final class OnlyOnceLoggingDenyMeterFilter implements MeterFilter { private final Supplier message; public OnlyOnceLoggingDenyMeterFilter(Supplier message) { - Assert.notNull(message, "Message must not be null"); + Assert.notNull(message, "'message' must not be null"); this.message = message; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/PropertiesMeterFilter.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/PropertiesMeterFilter.java index bf506756478a..89db89f76aa4 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/PropertiesMeterFilter.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/PropertiesMeterFilter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -51,7 +51,7 @@ public class PropertiesMeterFilter implements MeterFilter { private final MeterFilter mapFilter; public PropertiesMeterFilter(MetricsProperties properties) { - Assert.notNull(properties, "Properties must not be null"); + Assert.notNull(properties, "'properties' must not be null"); this.properties = properties; this.mapFilter = createMapFilter(properties.getTags()); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/appoptics/AppOpticsProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/appoptics/AppOpticsProperties.java index 90367e5e24d2..cd68586399ea 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/appoptics/AppOpticsProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/appoptics/AppOpticsProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,7 +28,7 @@ * @author Stephane Nicoll * @since 2.1.0 */ -@ConfigurationProperties(prefix = "management.appoptics.metrics.export") +@ConfigurationProperties("management.appoptics.metrics.export") public class AppOpticsProperties extends StepRegistryProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/atlas/AtlasProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/atlas/AtlasProperties.java index 1d476f0790ad..52cd8b3b1852 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/atlas/AtlasProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/atlas/AtlasProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,7 +28,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.atlas.metrics.export") +@ConfigurationProperties("management.atlas.metrics.export") public class AtlasProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/datadog/DatadogProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/datadog/DatadogProperties.java index b60f60d49465..63e71577a8ac 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/datadog/DatadogProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/datadog/DatadogProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.datadog.metrics.export") +@ConfigurationProperties("management.datadog.metrics.export") public class DatadogProperties extends StepRegistryProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/dynatrace/DynatraceProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/dynatrace/DynatraceProperties.java index 189ef09a86b5..2ffbe5ba9087 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/dynatrace/DynatraceProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/dynatrace/DynatraceProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Georg Pirklbauer * @since 2.1.0 */ -@ConfigurationProperties(prefix = "management.dynatrace.metrics.export") +@ConfigurationProperties("management.dynatrace.metrics.export") public class DynatraceProperties extends StepRegistryProperties { private final V1 v1 = new V1(); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/elastic/ElasticProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/elastic/ElasticProperties.java index d26b143b0063..a0c01a7d9875 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/elastic/ElasticProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/elastic/ElasticProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Andy Wilkinson * @since 2.1.0 */ -@ConfigurationProperties(prefix = "management.elastic.metrics.export") +@ConfigurationProperties("management.elastic.metrics.export") public class ElasticProperties extends StepRegistryProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/ganglia/GangliaProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/ganglia/GangliaProperties.java index 92bc9cb36487..2ac5a08bc35e 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/ganglia/GangliaProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/ganglia/GangliaProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.ganglia.metrics.export") +@ConfigurationProperties("management.ganglia.metrics.export") public class GangliaProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/graphite/GraphiteProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/graphite/GraphiteProperties.java index 47ed9e7ef82d..24c3e71bd10f 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/graphite/GraphiteProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/graphite/GraphiteProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.graphite.metrics.export") +@ConfigurationProperties("management.graphite.metrics.export") public class GraphiteProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/humio/HumioProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/humio/HumioProperties.java index a65930330e51..c61e7c8c05de 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/humio/HumioProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/humio/HumioProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,7 +30,7 @@ * @author Andy Wilkinson * @since 2.1.0 */ -@ConfigurationProperties(prefix = "management.humio.metrics.export") +@ConfigurationProperties("management.humio.metrics.export") public class HumioProperties extends StepRegistryProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/influx/InfluxProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/influx/InfluxProperties.java index 8cfb61c8d6a8..3888e295837e 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/influx/InfluxProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/influx/InfluxProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,7 +30,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.influx.metrics.export") +@ConfigurationProperties("management.influx.metrics.export") public class InfluxProperties extends StepRegistryProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/jmx/JmxProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/jmx/JmxProperties.java index 3e3d19db8c0e..e36b345791c5 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/jmx/JmxProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/jmx/JmxProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,7 +28,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.jmx.metrics.export") +@ConfigurationProperties("management.jmx.metrics.export") public class JmxProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/kairos/KairosProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/kairos/KairosProperties.java index 2206ad973d60..624dea85d793 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/kairos/KairosProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/kairos/KairosProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Stephane Nicoll * @since 2.1.0 */ -@ConfigurationProperties(prefix = "management.kairos.metrics.export") +@ConfigurationProperties("management.kairos.metrics.export") public class KairosProperties extends StepRegistryProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/newrelic/NewRelicProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/newrelic/NewRelicProperties.java index 965bc5b17a64..5c9283717cb2 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/newrelic/NewRelicProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/newrelic/NewRelicProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ * @author Neil Powell * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.newrelic.metrics.export") +@ConfigurationProperties("management.newrelic.metrics.export") public class NewRelicProperties extends StepRegistryProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsProperties.java index 257a8ce3ef0e..bcda23c88c65 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,6 @@ import org.springframework.boot.actuate.autoconfigure.metrics.export.properties.StepRegistryProperties; import org.springframework.boot.context.properties.ConfigurationProperties; -import org.springframework.boot.context.properties.DeprecatedConfigurationProperty; /** * {@link ConfigurationProperties @ConfigurationProperties} for configuring OTLP metrics @@ -34,13 +33,13 @@ * @author Jonatan Ivanov * @since 3.4.0 */ -@ConfigurationProperties(prefix = "management.otlp.metrics.export") +@ConfigurationProperties("management.otlp.metrics.export") public class OtlpMetricsProperties extends StepRegistryProperties { /** * URI of the OTLP server. */ - private String url = "http://localhost:4318/v1/metrics"; + private String url; /** * Aggregation temporality of sums. It defines the way additive values are expressed. @@ -48,11 +47,6 @@ public class OtlpMetricsProperties extends StepRegistryProperties { */ private AggregationTemporality aggregationTemporality = AggregationTemporality.CUMULATIVE; - /** - * Monitored resource's attributes. - */ - private Map resourceAttributes; - /** * Headers for the exported metrics. */ @@ -95,17 +89,6 @@ public void setAggregationTemporality(AggregationTemporality aggregationTemporal this.aggregationTemporality = aggregationTemporality; } - @Deprecated(since = "3.2.0", forRemoval = true) - @DeprecatedConfigurationProperty(replacement = "management.opentelemetry.resource-attributes", since = "3.2.0") - public Map getResourceAttributes() { - return this.resourceAttributes; - } - - @Deprecated(since = "3.2.0", forRemoval = true) - public void setResourceAttributes(Map resourceAttributes) { - this.resourceAttributes = resourceAttributes; - } - public Map getHeaders() { return this.headers; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesConfigAdapter.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesConfigAdapter.java index b6268dee51d2..8ad40855835f 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesConfigAdapter.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesConfigAdapter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,7 +17,7 @@ package org.springframework.boot.actuate.autoconfigure.metrics.export.otlp; import java.util.Collections; -import java.util.HashMap; +import java.util.LinkedHashMap; import java.util.Map; import java.util.concurrent.TimeUnit; @@ -27,9 +27,8 @@ import org.springframework.boot.actuate.autoconfigure.metrics.export.properties.StepRegistryPropertiesConfigAdapter; import org.springframework.boot.actuate.autoconfigure.opentelemetry.OpenTelemetryProperties; +import org.springframework.boot.actuate.autoconfigure.opentelemetry.OpenTelemetryResourceAttributes; import org.springframework.core.env.Environment; -import org.springframework.util.CollectionUtils; -import org.springframework.util.StringUtils; /** * Adapter to convert {@link OtlpMetricsProperties} to an {@link OtlpConfig}. @@ -41,11 +40,6 @@ class OtlpMetricsPropertiesConfigAdapter extends StepRegistryPropertiesConfigAdapter implements OtlpConfig { - /** - * Default value for application name if {@code spring.application.name} is not set. - */ - private static final String DEFAULT_APPLICATION_NAME = "unknown_service"; - private final OpenTelemetryProperties openTelemetryProperties; private final OtlpMetricsConnectionDetails connectionDetails; @@ -77,23 +71,11 @@ public AggregationTemporality aggregationTemporality() { } @Override - @SuppressWarnings("removal") public Map resourceAttributes() { - Map resourceAttributes = this.openTelemetryProperties.getResourceAttributes(); - Map result = new HashMap<>((!CollectionUtils.isEmpty(resourceAttributes)) ? resourceAttributes - : get(OtlpMetricsProperties::getResourceAttributes, OtlpConfig.super::resourceAttributes)); - result.computeIfAbsent("service.name", (key) -> getApplicationName()); - result.computeIfAbsent("service.group", (key) -> getApplicationGroup()); - return Collections.unmodifiableMap(result); - } - - private String getApplicationName() { - return this.environment.getProperty("spring.application.name", DEFAULT_APPLICATION_NAME); - } - - private String getApplicationGroup() { - String applicationGroup = this.environment.getProperty("spring.application.group"); - return (StringUtils.hasLength(applicationGroup)) ? applicationGroup : null; + Map resourceAttributes = new LinkedHashMap<>(); + new OpenTelemetryResourceAttributes(this.environment, this.openTelemetryProperties.getResourceAttributes()) + .applyTo(resourceAttributes::put); + return Collections.unmodifiableMap(resourceAttributes); } @Override diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusMetricsExportAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusMetricsExportAutoConfiguration.java index 3b1f705b8aa0..622ad8464a98 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusMetricsExportAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusMetricsExportAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,10 @@ import io.micrometer.core.instrument.Clock; import io.micrometer.prometheusmetrics.PrometheusConfig; import io.micrometer.prometheusmetrics.PrometheusMeterRegistry; +import io.prometheus.metrics.exporter.pushgateway.Format; +import io.prometheus.metrics.exporter.pushgateway.PushGateway; +import io.prometheus.metrics.exporter.pushgateway.PushGateway.Builder; +import io.prometheus.metrics.exporter.pushgateway.Scheme; import io.prometheus.metrics.model.registry.PrometheusRegistry; import io.prometheus.metrics.tracer.common.SpanContext; @@ -28,15 +32,20 @@ import org.springframework.boot.actuate.autoconfigure.metrics.MetricsAutoConfiguration; import org.springframework.boot.actuate.autoconfigure.metrics.export.ConditionalOnEnabledMetricsExport; import org.springframework.boot.actuate.autoconfigure.metrics.export.simple.SimpleMetricsExportAutoConfiguration; +import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusPushGatewayManager; import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusScrapeEndpoint; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.boot.context.properties.source.MutuallyExclusiveConfigurationPropertiesException; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.core.env.Environment; +import org.springframework.util.StringUtils; /** * {@link EnableAutoConfiguration Auto-configuration} for exporting metrics to Prometheus. @@ -78,10 +87,8 @@ PrometheusRegistry prometheusRegistry() { @ConditionalOnAvailableEndpoint(PrometheusScrapeEndpoint.class) static class PrometheusScrapeEndpointConfiguration { - @SuppressWarnings("removal") @Bean - @ConditionalOnMissingBean({ PrometheusScrapeEndpoint.class, - org.springframework.boot.actuate.metrics.export.prometheus.PrometheusSimpleclientScrapeEndpoint.class }) + @ConditionalOnMissingBean PrometheusScrapeEndpoint prometheusEndpoint(PrometheusRegistry prometheusRegistry, PrometheusConfig prometheusConfig) { return new PrometheusScrapeEndpoint(prometheusRegistry, prometheusConfig.prometheusProperties()); @@ -89,4 +96,73 @@ PrometheusScrapeEndpoint prometheusEndpoint(PrometheusRegistry prometheusRegistr } + /** + * Configuration for Prometheus + * Pushgateway. + */ + @Configuration(proxyBeanMethods = false) + @ConditionalOnClass(PushGateway.class) + @ConditionalOnBooleanProperty("management.prometheus.metrics.export.pushgateway.enabled") + static class PrometheusPushGatewayConfiguration { + + /** + * The fallback job name. We use 'spring' since there's a history of Prometheus + * Spring integration defaulting to that name from when Prometheus integration + * didn't exist in Spring itself. + */ + private static final String FALLBACK_JOB = "spring"; + + @Bean + @ConditionalOnMissingBean + PrometheusPushGatewayManager prometheusPushGatewayManager(PrometheusRegistry registry, + PrometheusProperties prometheusProperties, Environment environment) { + PrometheusProperties.Pushgateway properties = prometheusProperties.getPushgateway(); + PushGateway pushGateway = initializePushGateway(registry, properties, environment); + return new PrometheusPushGatewayManager(pushGateway, properties.getPushRate(), + properties.getShutdownOperation()); + } + + private PushGateway initializePushGateway(PrometheusRegistry registry, + PrometheusProperties.Pushgateway properties, Environment environment) { + Builder builder = PushGateway.builder() + .address(properties.getAddress()) + .scheme(scheme(properties)) + .format(format(properties)) + .job(getJob(properties, environment)) + .registry(registry); + MutuallyExclusiveConfigurationPropertiesException.throwIfMultipleNonNullValuesIn((entries) -> { + entries.put("management.prometheus.metrics.export.pushgateway.token", properties.getToken()); + entries.put("management.prometheus.metrics.export.pushgateway.username", properties.getUsername()); + }); + if (StringUtils.hasText(properties.getToken())) { + builder.bearerToken(properties.getToken()); + } + else if (StringUtils.hasText(properties.getUsername())) { + builder.basicAuth(properties.getUsername(), properties.getPassword()); + } + properties.getGroupingKey().forEach(builder::groupingKey); + return builder.build(); + } + + private Scheme scheme(PrometheusProperties.Pushgateway properties) { + return switch (properties.getScheme()) { + case HTTP -> Scheme.HTTP; + case HTTPS -> Scheme.HTTPS; + }; + } + + private Format format(PrometheusProperties.Pushgateway properties) { + return switch (properties.getFormat()) { + case PROTOBUF -> Format.PROMETHEUS_PROTOBUF; + case TEXT -> Format.PROMETHEUS_TEXT; + }; + } + + private String getJob(PrometheusProperties.Pushgateway properties, Environment environment) { + String job = properties.getJob(); + return (job != null) ? job : environment.getProperty("spring.application.name", FALLBACK_JOB); + } + + } + } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusProperties.java index c69429a48a96..bf1533f8be52 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,7 +22,6 @@ import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusPushGatewayManager.ShutdownOperation; import org.springframework.boot.context.properties.ConfigurationProperties; -import org.springframework.boot.context.properties.DeprecatedConfigurationProperty; /** * {@link ConfigurationProperties @ConfigurationProperties} for configuring metrics export @@ -32,7 +31,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.prometheus.metrics.export") +@ConfigurationProperties("management.prometheus.metrics.export") public class PrometheusProperties { /** @@ -52,13 +51,6 @@ public class PrometheusProperties { */ private final Pushgateway pushgateway = new Pushgateway(); - /** - * Histogram type for backing DistributionSummary and Timer. - * @deprecated since 3.3.0 for removal in 3.5.0 - */ - @Deprecated(since = "3.3.0", forRemoval = true) - private HistogramFlavor histogramFlavor = HistogramFlavor.Prometheus; - /** * Additional properties to pass to the Prometheus client. */ @@ -77,17 +69,6 @@ public void setDescriptions(boolean descriptions) { this.descriptions = descriptions; } - @Deprecated(since = "3.3.0", forRemoval = true) - @DeprecatedConfigurationProperty(since = "3.3.0", - reason = "No longer supported. Works only when using the Prometheus simpleclient.") - public HistogramFlavor getHistogramFlavor() { - return this.histogramFlavor; - } - - public void setHistogramFlavor(HistogramFlavor histogramFlavor) { - this.histogramFlavor = histogramFlavor; - } - public Duration getStep() { return this.step; } @@ -123,9 +104,14 @@ public static class Pushgateway { private Boolean enabled = false; /** - * Base URL for the Pushgateway. + * Address (host:port) for the Pushgateway. + */ + private String address = "localhost:9091"; + + /** + * Scheme to use when pushing metrics. */ - private String baseUrl = "http://localhost:9091"; + private Scheme scheme = Scheme.HTTP; /** * Login user of the Prometheus Pushgateway. @@ -137,6 +123,16 @@ public static class Pushgateway { */ private String password; + /** + * Token to use for authentication with the Prometheus Pushgateway. + */ + private String token; + + /** + * Format to use when pushing metrics. + */ + private Format format = Format.PROTOBUF; + /** * Frequency with which to push metrics. */ @@ -165,12 +161,12 @@ public void setEnabled(Boolean enabled) { this.enabled = enabled; } - public String getBaseUrl() { - return this.baseUrl; + public String getAddress() { + return this.address; } - public void setBaseUrl(String baseUrl) { - this.baseUrl = baseUrl; + public void setAddress(String address) { + this.address = address; } public String getUsername() { @@ -221,17 +217,57 @@ public void setShutdownOperation(ShutdownOperation shutdownOperation) { this.shutdownOperation = shutdownOperation; } - } + public Scheme getScheme() { + return this.scheme; + } - /** - * Prometheus Histogram flavor. - * - * @deprecated since 3.3.0 for removal in 3.5.0 - */ - @Deprecated(since = "3.3.0", forRemoval = true) - public enum HistogramFlavor { + public void setScheme(Scheme scheme) { + this.scheme = scheme; + } + + public String getToken() { + return this.token; + } + + public void setToken(String token) { + this.token = token; + } + + public Format getFormat() { + return this.format; + } + + public void setFormat(Format format) { + this.format = format; + } - Prometheus, VictoriaMetrics + public enum Format { + + /** + * Push metrics in text format. + */ + TEXT, + + /** + * Push metrics in protobuf format. + */ + PROTOBUF + + } + + public enum Scheme { + + /** + * Use HTTP to push metrics. + */ + HTTP, + + /** + * Use HTTPS to push metrics. + */ + HTTPS + + } } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientMetricsExportAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientMetricsExportAutoConfiguration.java deleted file mode 100644 index 25c94dd52912..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientMetricsExportAutoConfiguration.java +++ /dev/null @@ -1,162 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus; - -import java.net.MalformedURLException; -import java.net.URL; -import java.time.Duration; -import java.util.Map; - -import io.micrometer.core.instrument.Clock; -import io.prometheus.client.CollectorRegistry; -import io.prometheus.client.exemplars.DefaultExemplarSampler; -import io.prometheus.client.exemplars.ExemplarSampler; -import io.prometheus.client.exemplars.tracer.common.SpanContextSupplier; -import io.prometheus.client.exporter.BasicAuthHttpConnectionFactory; -import io.prometheus.client.exporter.PushGateway; - -import org.springframework.beans.factory.ObjectProvider; -import org.springframework.boot.actuate.autoconfigure.endpoint.condition.ConditionalOnAvailableEndpoint; -import org.springframework.boot.actuate.autoconfigure.metrics.CompositeMeterRegistryAutoConfiguration; -import org.springframework.boot.actuate.autoconfigure.metrics.MetricsAutoConfiguration; -import org.springframework.boot.actuate.autoconfigure.metrics.export.ConditionalOnEnabledMetricsExport; -import org.springframework.boot.actuate.autoconfigure.metrics.export.simple.SimpleMetricsExportAutoConfiguration; -import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusPushGatewayManager; -import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusPushGatewayManager.ShutdownOperation; -import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusScrapeEndpoint; -import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusSimpleclientScrapeEndpoint; -import org.springframework.boot.autoconfigure.AutoConfiguration; -import org.springframework.boot.autoconfigure.EnableAutoConfiguration; -import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; -import org.springframework.boot.context.properties.EnableConfigurationProperties; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.core.env.Environment; -import org.springframework.util.StringUtils; - -/** - * {@link EnableAutoConfiguration Auto-configuration} for exporting metrics to Prometheus - * with the Prometheus simpleclient. - * - * @author Jon Schneider - * @author David J. M. Karlsen - * @author Jonatan Ivanov - * @since 2.0.0 - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of - * {@link PrometheusMetricsExportAutoConfiguration} - */ -@Deprecated(since = "3.3.0", forRemoval = true) -@AutoConfiguration( - before = { CompositeMeterRegistryAutoConfiguration.class, SimpleMetricsExportAutoConfiguration.class }, - after = { MetricsAutoConfiguration.class, PrometheusMetricsExportAutoConfiguration.class }) -@ConditionalOnBean(Clock.class) -@ConditionalOnClass(io.micrometer.prometheus.PrometheusMeterRegistry.class) -@ConditionalOnEnabledMetricsExport("prometheus") -@EnableConfigurationProperties(PrometheusProperties.class) -public class PrometheusSimpleclientMetricsExportAutoConfiguration { - - @Bean - @ConditionalOnMissingBean - io.micrometer.prometheus.PrometheusConfig simpleclientPrometheusConfig(PrometheusProperties prometheusProperties) { - return new PrometheusSimpleclientPropertiesConfigAdapter(prometheusProperties); - } - - @Bean - @ConditionalOnMissingBean - io.micrometer.prometheus.PrometheusMeterRegistry simpleclientPrometheusMeterRegistry( - io.micrometer.prometheus.PrometheusConfig prometheusConfig, CollectorRegistry collectorRegistry, - Clock clock, ObjectProvider exemplarSamplerProvider) { - return new io.micrometer.prometheus.PrometheusMeterRegistry(prometheusConfig, collectorRegistry, clock, - exemplarSamplerProvider.getIfAvailable()); - } - - @Bean - @ConditionalOnMissingBean - CollectorRegistry collectorRegistry() { - return new CollectorRegistry(true); - } - - @Bean - @ConditionalOnMissingBean(ExemplarSampler.class) - @ConditionalOnBean(SpanContextSupplier.class) - DefaultExemplarSampler exemplarSampler(SpanContextSupplier spanContextSupplier) { - return new DefaultExemplarSampler(spanContextSupplier); - } - - @SuppressWarnings("removal") - @Configuration(proxyBeanMethods = false) - @ConditionalOnAvailableEndpoint(PrometheusSimpleclientScrapeEndpoint.class) - static class PrometheusScrapeEndpointConfiguration { - - @Bean - @ConditionalOnMissingBean({ PrometheusSimpleclientScrapeEndpoint.class, PrometheusScrapeEndpoint.class }) - PrometheusSimpleclientScrapeEndpoint prometheusEndpoint(CollectorRegistry collectorRegistry) { - return new PrometheusSimpleclientScrapeEndpoint(collectorRegistry); - } - - } - - /** - * Configuration for Prometheus - * Pushgateway. - */ - @Configuration(proxyBeanMethods = false) - @ConditionalOnClass(PushGateway.class) - @ConditionalOnProperty(prefix = "management.prometheus.metrics.export.pushgateway", name = "enabled") - static class PrometheusPushGatewayConfiguration { - - /** - * The fallback job name. We use 'spring' since there's a history of Prometheus - * spring integration defaulting to that name from when Prometheus integration - * didn't exist in Spring itself. - */ - private static final String FALLBACK_JOB = "spring"; - - @Bean - @ConditionalOnMissingBean - PrometheusPushGatewayManager prometheusPushGatewayManager(CollectorRegistry collectorRegistry, - PrometheusProperties prometheusProperties, Environment environment) throws MalformedURLException { - PrometheusProperties.Pushgateway properties = prometheusProperties.getPushgateway(); - Duration pushRate = properties.getPushRate(); - String job = getJob(properties, environment); - Map groupingKey = properties.getGroupingKey(); - ShutdownOperation shutdownOperation = properties.getShutdownOperation(); - PushGateway pushGateway = initializePushGateway(properties.getBaseUrl()); - if (StringUtils.hasText(properties.getUsername())) { - pushGateway.setConnectionFactory( - new BasicAuthHttpConnectionFactory(properties.getUsername(), properties.getPassword())); - } - return new PrometheusPushGatewayManager(pushGateway, collectorRegistry, pushRate, job, groupingKey, - shutdownOperation); - } - - private PushGateway initializePushGateway(String url) throws MalformedURLException { - return new PushGateway(new URL(url)); - } - - private String getJob(PrometheusProperties.Pushgateway properties, Environment environment) { - String job = properties.getJob(); - job = (job != null) ? job : environment.getProperty("spring.application.name"); - return (job != null) ? job : FALLBACK_JOB; - } - - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientPropertiesConfigAdapter.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientPropertiesConfigAdapter.java deleted file mode 100644 index d685f1bdcf31..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientPropertiesConfigAdapter.java +++ /dev/null @@ -1,71 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus; - -import java.time.Duration; - -import org.springframework.boot.actuate.autoconfigure.metrics.export.properties.PropertiesConfigAdapter; - -/** - * Adapter to convert {@link PrometheusProperties} to a - * {@link io.micrometer.prometheus.PrometheusConfig}. - * - * @author Jon Schneider - * @author Phillip Webb - */ -@SuppressWarnings({ "deprecation", "removal" }) -class PrometheusSimpleclientPropertiesConfigAdapter extends PropertiesConfigAdapter - implements io.micrometer.prometheus.PrometheusConfig { - - PrometheusSimpleclientPropertiesConfigAdapter(PrometheusProperties properties) { - super(properties); - } - - @Override - public String prefix() { - return "management.prometheus.metrics.export"; - } - - @Override - public String get(String key) { - return null; - } - - @Override - public boolean descriptions() { - return get(PrometheusProperties::isDescriptions, io.micrometer.prometheus.PrometheusConfig.super::descriptions); - } - - @Override - public io.micrometer.prometheus.HistogramFlavor histogramFlavor() { - return get(PrometheusSimpleclientPropertiesConfigAdapter::mapToMicrometerHistogramFlavor, - io.micrometer.prometheus.PrometheusConfig.super::histogramFlavor); - } - - static io.micrometer.prometheus.HistogramFlavor mapToMicrometerHistogramFlavor(PrometheusProperties properties) { - return switch (properties.getHistogramFlavor()) { - case Prometheus -> io.micrometer.prometheus.HistogramFlavor.Prometheus; - case VictoriaMetrics -> io.micrometer.prometheus.HistogramFlavor.VictoriaMetrics; - }; - } - - @Override - public Duration step() { - return get(PrometheusProperties::getStep, io.micrometer.prometheus.PrometheusConfig.super::step); - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/properties/PropertiesConfigAdapter.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/properties/PropertiesConfigAdapter.java index 4cc34a988868..8b0371139aa3 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/properties/PropertiesConfigAdapter.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/properties/PropertiesConfigAdapter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ public class PropertiesConfigAdapter { * @param properties the source properties */ public PropertiesConfigAdapter(T properties) { - Assert.notNull(properties, "Properties must not be null"); + Assert.notNull(properties, "'properties' must not be null"); this.properties = properties; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxMetricsExportAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxMetricsExportAutoConfiguration.java index 326c8248b27d..7be7440b14b1 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxMetricsExportAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxMetricsExportAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,6 +38,7 @@ * @author Jon Schneider * @author Andy Wilkinson * @since 2.0.0 + * @deprecated since 3.5.0 for removal in 4.0.0 */ @AutoConfiguration( before = { CompositeMeterRegistryAutoConfiguration.class, SimpleMetricsExportAutoConfiguration.class }, @@ -46,6 +47,8 @@ @ConditionalOnClass(SignalFxMeterRegistry.class) @ConditionalOnEnabledMetricsExport("signalfx") @EnableConfigurationProperties(SignalFxProperties.class) +@Deprecated(since = "3.5.0", forRemoval = true) +@SuppressWarnings("removal") public class SignalFxMetricsExportAutoConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxProperties.java index 0b9555b9aecb..b13353dc957b 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,6 +20,7 @@ import org.springframework.boot.actuate.autoconfigure.metrics.export.properties.StepRegistryProperties; import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.context.properties.DeprecatedConfigurationProperty; /** * {@link ConfigurationProperties @ConfigurationProperties} for configuring metrics export @@ -29,8 +30,10 @@ * @author Andy Wilkinson * @author Stephane Nicoll * @since 2.0.0 + * @deprecated since 3.5.0 for removal in 4.0.0 */ -@ConfigurationProperties(prefix = "management.signalfx.metrics.export") +@ConfigurationProperties("management.signalfx.metrics.export") +@Deprecated(since = "3.5.0", forRemoval = true) public class SignalFxProperties extends StepRegistryProperties { /** @@ -60,47 +63,115 @@ public class SignalFxProperties extends StepRegistryProperties { private HistogramType publishedHistogramType = HistogramType.DEFAULT; @Override + @DeprecatedConfigurationProperty(since = "3.5.0", reason = "Deprecated in Micrometer 1.15.0") + @Deprecated(since = "3.5.0", forRemoval = true) public Duration getStep() { return this.step; } @Override + @Deprecated(since = "3.5.0", forRemoval = true) public void setStep(Duration step) { this.step = step; } + @DeprecatedConfigurationProperty(since = "3.5.0", reason = "Deprecated in Micrometer 1.15.0") + @Deprecated(since = "3.5.0", forRemoval = true) public String getAccessToken() { return this.accessToken; } + @Deprecated(since = "3.5.0", forRemoval = true) public void setAccessToken(String accessToken) { this.accessToken = accessToken; } + @DeprecatedConfigurationProperty(since = "3.5.0", reason = "Deprecated in Micrometer 1.15.0") + @Deprecated(since = "3.5.0", forRemoval = true) public String getUri() { return this.uri; } + @Deprecated(since = "3.5.0", forRemoval = true) public void setUri(String uri) { this.uri = uri; } + @DeprecatedConfigurationProperty(since = "3.5.0", reason = "Deprecated in Micrometer 1.15.0") + @Deprecated(since = "3.5.0", forRemoval = true) public String getSource() { return this.source; } + @Deprecated(since = "3.5.0", forRemoval = true) public void setSource(String source) { this.source = source; } + @DeprecatedConfigurationProperty(since = "3.5.0", reason = "Deprecated in Micrometer 1.15.0") + @Deprecated(since = "3.5.0", forRemoval = true) public HistogramType getPublishedHistogramType() { return this.publishedHistogramType; } + @Deprecated(since = "3.5.0", forRemoval = true) public void setPublishedHistogramType(HistogramType publishedHistogramType) { this.publishedHistogramType = publishedHistogramType; } + @Override + @DeprecatedConfigurationProperty(since = "3.5.0", reason = "Deprecated in Micrometer 1.15.0") + @Deprecated(since = "3.5.0", forRemoval = true) + public boolean isEnabled() { + return super.isEnabled(); + } + + @Override + @Deprecated(since = "3.5.0", forRemoval = true) + public void setEnabled(boolean enabled) { + super.setEnabled(enabled); + } + + @Override + @DeprecatedConfigurationProperty(since = "3.5.0", reason = "Deprecated in Micrometer 1.15.0") + @Deprecated(since = "3.5.0", forRemoval = true) + public Duration getConnectTimeout() { + return super.getConnectTimeout(); + } + + @Override + @Deprecated(since = "3.5.0", forRemoval = true) + public void setConnectTimeout(Duration connectTimeout) { + super.setConnectTimeout(connectTimeout); + } + + @Override + @DeprecatedConfigurationProperty(since = "3.5.0", reason = "Deprecated in Micrometer 1.15.0") + @Deprecated(since = "3.5.0", forRemoval = true) + public Duration getReadTimeout() { + return super.getReadTimeout(); + } + + @Override + @Deprecated(since = "3.5.0", forRemoval = true) + public void setReadTimeout(Duration readTimeout) { + super.setReadTimeout(readTimeout); + } + + @Override + @DeprecatedConfigurationProperty(since = "3.5.0", reason = "Deprecated in Micrometer 1.15.0") + @Deprecated(since = "3.5.0", forRemoval = true) + public Integer getBatchSize() { + return super.getBatchSize(); + } + + @Override + @Deprecated(since = "3.5.0", forRemoval = true) + public void setBatchSize(Integer batchSize) { + super.setBatchSize(batchSize); + } + + @Deprecated(since = "3.5.0", forRemoval = true) public enum HistogramType { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesConfigAdapter.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesConfigAdapter.java index 754e3cb7c4df..c737565e9d93 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesConfigAdapter.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesConfigAdapter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,14 +19,16 @@ import io.micrometer.signalfx.SignalFxConfig; import org.springframework.boot.actuate.autoconfigure.metrics.export.properties.StepRegistryPropertiesConfigAdapter; -import org.springframework.boot.actuate.autoconfigure.metrics.export.signalfx.SignalFxProperties.HistogramType; /** * Adapter to convert {@link SignalFxProperties} to a {@link SignalFxConfig}. * * @author Jon Schneider * @since 2.0.0 + * @deprecated since 3.5.0 for removal in 4.0.0 */ +@Deprecated(since = "3.5.0", forRemoval = true) +@SuppressWarnings("removal") public class SignalFxPropertiesConfigAdapter extends StepRegistryPropertiesConfigAdapter implements SignalFxConfig { @@ -61,7 +63,7 @@ public boolean publishCumulativeHistogram() { } private boolean isPublishCumulativeHistogram(SignalFxProperties properties) { - return HistogramType.CUMULATIVE == properties.getPublishedHistogramType(); + return SignalFxProperties.HistogramType.CUMULATIVE == properties.getPublishedHistogramType(); } @Override @@ -70,7 +72,7 @@ public boolean publishDeltaHistogram() { } private boolean isPublishDeltaHistogram(SignalFxProperties properties) { - return HistogramType.DELTA == properties.getPublishedHistogramType(); + return SignalFxProperties.HistogramType.DELTA == properties.getPublishedHistogramType(); } } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/simple/SimpleProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/simple/SimpleProperties.java index 26715be4997f..59a6d0947569 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/simple/SimpleProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/simple/SimpleProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.simple.metrics.export") +@ConfigurationProperties("management.simple.metrics.export") public class SimpleProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/stackdriver/StackdriverProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/stackdriver/StackdriverProperties.java index 4557f3f5390c..996a18e227b3 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/stackdriver/StackdriverProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/stackdriver/StackdriverProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Stephane Nicoll * @since 2.3.0 */ -@ConfigurationProperties(prefix = "management.stackdriver.metrics.export") +@ConfigurationProperties("management.stackdriver.metrics.export") public class StackdriverProperties extends StepRegistryProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/statsd/StatsdProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/statsd/StatsdProperties.java index ca6ba6cf6146..8c9c8c39e138 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/statsd/StatsdProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/export/statsd/StatsdProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.statsd.metrics.export") +@ConfigurationProperties("management.statsd.metrics.export") public class StatsdProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/mongo/MongoMetricsAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/mongo/MongoMetricsAutoConfiguration.java index e900d07674e4..b45d3021fe9f 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/mongo/MongoMetricsAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/mongo/MongoMetricsAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,9 +30,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.mongo.MongoAutoConfiguration; import org.springframework.boot.autoconfigure.mongo.MongoClientSettingsBuilderCustomizer; import org.springframework.context.annotation.Bean; @@ -51,8 +51,7 @@ public class MongoMetricsAutoConfiguration { @ConditionalOnClass(MongoMetricsCommandListener.class) - @ConditionalOnProperty(name = "management.metrics.mongo.command.enabled", havingValue = "true", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "management.metrics.mongo.command.enabled", matchIfMissing = true) static class MongoCommandMetricsConfiguration { @Bean @@ -77,8 +76,7 @@ MongoClientSettingsBuilderCustomizer mongoMetricsCommandListenerClientSettingsBu } @ConditionalOnClass(MongoMetricsConnectionPoolListener.class) - @ConditionalOnProperty(name = "management.metrics.mongo.connectionpool.enabled", havingValue = "true", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "management.metrics.mongo.connectionpool.enabled", matchIfMissing = true) static class MongoConnectionPoolMetricsConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/web/jetty/JettyMetricsAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/web/jetty/JettyMetricsAutoConfiguration.java index 4bd6eb69bfcd..bb1191c570c2 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/web/jetty/JettyMetricsAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/metrics/web/jetty/JettyMetricsAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,9 +29,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.context.annotation.Bean; @@ -62,7 +62,7 @@ public JettyConnectionMetricsBinder jettyConnectionMetricsBinder(MeterRegistry m @Bean @ConditionalOnMissingBean({ JettySslHandshakeMetrics.class, JettySslHandshakeMetricsBinder.class }) - @ConditionalOnProperty(name = "server.ssl.enabled", havingValue = "true") + @ConditionalOnBooleanProperty("server.ssl.enabled") public JettySslHandshakeMetricsBinder jettySslHandshakeMetricsBinder(MeterRegistry meterRegistry) { return new JettySslHandshakeMetricsBinder(meterRegistry); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/observation/ObservationProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/observation/ObservationProperties.java index 93d0e7152464..7662c04a7efd 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/observation/ObservationProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/observation/ObservationProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/observation/web/servlet/WebMvcObservationAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/observation/web/servlet/WebMvcObservationAutoConfiguration.java index 2b4aa96c3933..105b8592d32e 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/observation/web/servlet/WebMvcObservationAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/observation/web/servlet/WebMvcObservationAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryAutoConfiguration.java index 720e79142f61..c19a7c8a9579 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,8 +17,6 @@ package org.springframework.boot.actuate.autoconfigure.opentelemetry; import io.opentelemetry.api.OpenTelemetry; -import io.opentelemetry.api.common.AttributeKey; -import io.opentelemetry.api.common.Attributes; import io.opentelemetry.context.propagation.ContextPropagators; import io.opentelemetry.sdk.OpenTelemetrySdk; import io.opentelemetry.sdk.OpenTelemetrySdkBuilder; @@ -36,7 +34,6 @@ import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; import org.springframework.core.env.Environment; -import org.springframework.util.StringUtils; /** * {@link EnableAutoConfiguration Auto-configuration} for OpenTelemetry. @@ -49,15 +46,6 @@ @EnableConfigurationProperties(OpenTelemetryProperties.class) public class OpenTelemetryAutoConfiguration { - /** - * Default value for application name if {@code spring.application.name} is not set. - */ - private static final String DEFAULT_APPLICATION_NAME = "unknown_service"; - - private static final AttributeKey ATTRIBUTE_KEY_SERVICE_NAME = AttributeKey.stringKey("service.name"); - - private static final AttributeKey ATTRIBUTE_KEY_SERVICE_GROUP = AttributeKey.stringKey("service.group"); - @Bean @ConditionalOnMissingBean(OpenTelemetry.class) OpenTelemetrySdk openTelemetry(ObjectProvider tracerProvider, @@ -74,19 +62,13 @@ OpenTelemetrySdk openTelemetry(ObjectProvider tracerProvider, @Bean @ConditionalOnMissingBean Resource openTelemetryResource(Environment environment, OpenTelemetryProperties properties) { - String applicationName = environment.getProperty("spring.application.name", DEFAULT_APPLICATION_NAME); - String applicationGroup = environment.getProperty("spring.application.group"); - Resource resource = Resource.getDefault() - .merge(Resource.create(Attributes.of(ATTRIBUTE_KEY_SERVICE_NAME, applicationName))); - if (StringUtils.hasLength(applicationGroup)) { - resource = resource.merge(Resource.create(Attributes.of(ATTRIBUTE_KEY_SERVICE_GROUP, applicationGroup))); - } - return resource.merge(toResource(properties)); + Resource resource = Resource.getDefault(); + return resource.merge(toResource(environment, properties)); } - private static Resource toResource(OpenTelemetryProperties properties) { + private Resource toResource(Environment environment, OpenTelemetryProperties properties) { ResourceBuilder builder = Resource.builder(); - properties.getResourceAttributes().forEach(builder::put); + new OpenTelemetryResourceAttributes(environment, properties.getResourceAttributes()).applyTo(builder::put); return builder.build(); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryProperties.java index 4c973ecf578b..bc3589af4769 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ * @author Moritz Halbritter * @since 3.2.0 */ -@ConfigurationProperties(prefix = "management.opentelemetry") +@ConfigurationProperties("management.opentelemetry") public class OpenTelemetryProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryResourceAttributes.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryResourceAttributes.java new file mode 100644 index 000000000000..0acdf0fa0c3b --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryResourceAttributes.java @@ -0,0 +1,195 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.opentelemetry; + +import java.io.ByteArrayOutputStream; +import java.nio.charset.StandardCharsets; +import java.util.Collections; +import java.util.LinkedHashMap; +import java.util.Map; +import java.util.function.BiConsumer; +import java.util.function.Function; + +import org.springframework.core.env.Environment; +import org.springframework.util.Assert; +import org.springframework.util.StringUtils; + +/** + * {@link OpenTelemetryResourceAttributes} retrieves information from the + * {@code OTEL_RESOURCE_ATTRIBUTES} and {@code OTEL_SERVICE_NAME} environment variables + * and merges it with the resource attributes provided by the user. User-provided resource + * attributes take precedence. Additionally, {@code spring.application.*} related + * properties can be applied as defaults. + *

+ * OpenTelemetry + * Resource Specification + * + * @author Dmytro Nosan + * @since 3.5.0 + */ +public final class OpenTelemetryResourceAttributes { + + /** + * Default value for service name if {@code service.name} is not set. + */ + private static final String DEFAULT_SERVICE_NAME = "unknown_service"; + + private final Environment environment; + + private final Map resourceAttributes; + + private final Function getEnv; + + /** + * Creates a new instance of {@link OpenTelemetryResourceAttributes}. + * @param environment the environment + * @param resourceAttributes user-provided resource attributes to be used + */ + public OpenTelemetryResourceAttributes(Environment environment, Map resourceAttributes) { + this(environment, resourceAttributes, null); + } + + /** + * Creates a new {@link OpenTelemetryResourceAttributes} instance. + * @param environment the environment + * @param resourceAttributes user-provided resource attributes to be used + * @param getEnv a function to retrieve environment variables by name + */ + OpenTelemetryResourceAttributes(Environment environment, Map resourceAttributes, + Function getEnv) { + Assert.notNull(environment, "'environment' must not be null"); + this.environment = environment; + this.resourceAttributes = (resourceAttributes != null) ? resourceAttributes : Collections.emptyMap(); + this.getEnv = (getEnv != null) ? getEnv : System::getenv; + } + + /** + * Applies resource attributes to the provided {@link BiConsumer} after being combined + * from environment variables and user-defined resource attributes. + *

+ * If a key exists in both environment variables and user-defined resources, the value + * from the user-defined resource takes precedence, even if it is empty. + *

+ * Additionally, {@code spring.application.name} or {@code unknown_service} will be + * used as the default for {@code service.name}, and {@code spring.application.group} + * will serve as the default for {@code service.group} and {@code service.namespace}. + * @param consumer the {@link BiConsumer} to apply + */ + public void applyTo(BiConsumer consumer) { + Assert.notNull(consumer, "'consumer' must not be null"); + Map attributes = getResourceAttributesFromEnv(); + this.resourceAttributes.forEach((name, value) -> { + if (StringUtils.hasLength(name) && value != null) { + attributes.put(name, value); + } + }); + attributes.computeIfAbsent("service.name", (key) -> getApplicationName()); + attributes.computeIfAbsent("service.group", (key) -> getApplicationGroup()); + attributes.computeIfAbsent("service.namespace", (key) -> getServiceNamespace()); + attributes.forEach(consumer); + } + + private String getApplicationName() { + return this.environment.getProperty("spring.application.name", DEFAULT_SERVICE_NAME); + } + + /** + * Returns the application group. + * @return the application group + * @deprecated since 3.5.0 for removal in 4.0.0 + */ + @Deprecated(since = "3.5.0", forRemoval = true) + private String getApplicationGroup() { + String applicationGroup = this.environment.getProperty("spring.application.group"); + return (StringUtils.hasLength(applicationGroup)) ? applicationGroup : null; + } + + private String getServiceNamespace() { + return this.environment.getProperty("spring.application.group"); + } + + /** + * Parses resource attributes from the {@link System#getenv()}. This method fetches + * attributes defined in the {@code OTEL_RESOURCE_ATTRIBUTES} and + * {@code OTEL_SERVICE_NAME} environment variables and provides them as key-value + * pairs. + *

+ * If {@code service.name} is also provided in {@code OTEL_RESOURCE_ATTRIBUTES}, then + * {@code OTEL_SERVICE_NAME} takes precedence. + * @return resource attributes + */ + private Map getResourceAttributesFromEnv() { + Map attributes = new LinkedHashMap<>(); + for (String attribute : StringUtils.tokenizeToStringArray(getEnv("OTEL_RESOURCE_ATTRIBUTES"), ",")) { + int index = attribute.indexOf('='); + if (index > 0) { + String key = attribute.substring(0, index); + String value = attribute.substring(index + 1); + attributes.put(key.trim(), decode(value.trim())); + } + } + String otelServiceName = getEnv("OTEL_SERVICE_NAME"); + if (otelServiceName != null) { + attributes.put("service.name", otelServiceName); + } + return attributes; + } + + private String getEnv(String name) { + return this.getEnv.apply(name); + } + + /** + * Decodes a percent-encoded string. Converts sequences like '%HH' (where HH + * represents hexadecimal digits) back into their literal representations. + *

+ * Inspired by {@code org.apache.commons.codec.net.PercentCodec}. + * @param value value to decode + * @return the decoded string + */ + private static String decode(String value) { + if (value.indexOf('%') < 0) { + return value; + } + byte[] bytes = value.getBytes(StandardCharsets.UTF_8); + ByteArrayOutputStream bos = new ByteArrayOutputStream(bytes.length); + for (int i = 0; i < bytes.length; i++) { + byte b = bytes[i]; + if (b != '%') { + bos.write(b); + continue; + } + int u = decodeHex(bytes, i + 1); + int l = decodeHex(bytes, i + 2); + if (u >= 0 && l >= 0) { + bos.write((u << 4) + l); + } + else { + throw new IllegalArgumentException( + "Failed to decode percent-encoded characters at index %d in the value: '%s'".formatted(i, + value)); + } + i += 2; + } + return bos.toString(StandardCharsets.UTF_8); + } + + private static int decodeHex(byte[] bytes, int index) { + return (index < bytes.length) ? Character.digit(bytes[index], 16) : -1; + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequest.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequest.java index ac538c616c87..a889f8b18bfa 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequest.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequest.java @@ -41,6 +41,7 @@ import org.springframework.context.ApplicationContext; import org.springframework.core.annotation.MergedAnnotation; import org.springframework.core.annotation.MergedAnnotations; +import org.springframework.http.HttpMethod; import org.springframework.security.web.server.util.matcher.OrServerWebExchangeMatcher; import org.springframework.security.web.server.util.matcher.PathPatternParserServerWebExchangeMatcher; import org.springframework.security.web.server.util.matcher.ServerWebExchangeMatcher; @@ -57,6 +58,7 @@ * * @author Madhura Bhave * @author Phillip Webb + * @author Chris Bono * @since 2.0.0 */ public final class EndpointRequest { @@ -181,12 +183,14 @@ private ServerWebExchangeMatcher createDelegate(Supplier context) { protected abstract ServerWebExchangeMatcher createDelegate(C context); - protected final List getDelegateMatchers(Set paths) { - return paths.stream().map(this::getDelegateMatcher).collect(Collectors.toCollection(ArrayList::new)); + protected final List getDelegateMatchers(Set paths, HttpMethod httpMethod) { + return paths.stream() + .map((path) -> getDelegateMatcher(path, httpMethod)) + .collect(Collectors.toCollection(ArrayList::new)); } - private PathPatternParserServerWebExchangeMatcher getDelegateMatcher(String path) { - return new PathPatternParserServerWebExchangeMatcher(path + "/**"); + private PathPatternParserServerWebExchangeMatcher getDelegateMatcher(String path, HttpMethod httpMethod) { + return new PathPatternParserServerWebExchangeMatcher(path + "/**", httpMethod); } @Override @@ -263,39 +267,53 @@ public static final class EndpointServerWebExchangeMatcher extends AbstractWebEx private final boolean includeLinks; + private final HttpMethod httpMethod; + private EndpointServerWebExchangeMatcher(boolean includeLinks) { - this(Collections.emptyList(), Collections.emptyList(), includeLinks); + this(Collections.emptyList(), Collections.emptyList(), includeLinks, null); } private EndpointServerWebExchangeMatcher(Class[] endpoints, boolean includeLinks) { - this(Arrays.asList((Object[]) endpoints), Collections.emptyList(), includeLinks); + this(Arrays.asList((Object[]) endpoints), Collections.emptyList(), includeLinks, null); } private EndpointServerWebExchangeMatcher(String[] endpoints, boolean includeLinks) { - this(Arrays.asList((Object[]) endpoints), Collections.emptyList(), includeLinks); + this(Arrays.asList((Object[]) endpoints), Collections.emptyList(), includeLinks, null); } - private EndpointServerWebExchangeMatcher(List includes, List excludes, boolean includeLinks) { + private EndpointServerWebExchangeMatcher(List includes, List excludes, boolean includeLinks, + HttpMethod httpMethod) { super(PathMappedEndpoints.class); this.includes = includes; this.excludes = excludes; this.includeLinks = includeLinks; + this.httpMethod = httpMethod; } public EndpointServerWebExchangeMatcher excluding(Class... endpoints) { List excludes = new ArrayList<>(this.excludes); excludes.addAll(Arrays.asList((Object[]) endpoints)); - return new EndpointServerWebExchangeMatcher(this.includes, excludes, this.includeLinks); + return new EndpointServerWebExchangeMatcher(this.includes, excludes, this.includeLinks, null); } public EndpointServerWebExchangeMatcher excluding(String... endpoints) { List excludes = new ArrayList<>(this.excludes); excludes.addAll(Arrays.asList((Object[]) endpoints)); - return new EndpointServerWebExchangeMatcher(this.includes, excludes, this.includeLinks); + return new EndpointServerWebExchangeMatcher(this.includes, excludes, this.includeLinks, null); } public EndpointServerWebExchangeMatcher excludingLinks() { - return new EndpointServerWebExchangeMatcher(this.includes, this.excludes, false); + return new EndpointServerWebExchangeMatcher(this.includes, this.excludes, false, null); + } + + /** + * Restricts the matcher to only consider requests with a particular http method. + * @param httpMethod the http method to include + * @return a copy of the matcher further restricted to only match requests with + * the specified http method + */ + public EndpointServerWebExchangeMatcher withHttpMethod(HttpMethod httpMethod) { + return new EndpointServerWebExchangeMatcher(this.includes, this.excludes, this.includeLinks, httpMethod); } @Override @@ -306,7 +324,7 @@ protected ServerWebExchangeMatcher createDelegate(PathMappedEndpoints endpoints) } streamPaths(this.includes, endpoints).forEach(paths::add); streamPaths(this.excludes, endpoints).forEach(paths::remove); - List delegateMatchers = getDelegateMatchers(paths); + List delegateMatchers = getDelegateMatchers(paths, this.httpMethod); if (this.includeLinks && StringUtils.hasText(endpoints.getBasePath())) { delegateMatchers.add(new LinksServerWebExchangeMatcher()); } @@ -362,22 +380,37 @@ public static class AdditionalPathsEndpointServerWebExchangeMatcher private final List endpoints; + private final HttpMethod httpMethod; + AdditionalPathsEndpointServerWebExchangeMatcher(WebServerNamespace webServerNamespace, String... endpoints) { - this(webServerNamespace, Arrays.asList((Object[]) endpoints)); + this(webServerNamespace, Arrays.asList((Object[]) endpoints), null); } AdditionalPathsEndpointServerWebExchangeMatcher(WebServerNamespace webServerNamespace, Class... endpoints) { - this(webServerNamespace, Arrays.asList((Object[]) endpoints)); + this(webServerNamespace, Arrays.asList((Object[]) endpoints), null); } private AdditionalPathsEndpointServerWebExchangeMatcher(WebServerNamespace webServerNamespace, - List endpoints) { + List endpoints, HttpMethod httpMethod) { super(PathMappedEndpoints.class); Assert.notNull(webServerNamespace, "'webServerNamespace' must not be null"); Assert.notNull(endpoints, "'endpoints' must not be null"); Assert.notEmpty(endpoints, "'endpoints' must not be empty"); this.webServerNamespace = webServerNamespace; this.endpoints = endpoints; + this.httpMethod = httpMethod; + } + + /** + * Restricts the matcher to only consider requests with a particular HTTP method. + * @param httpMethod the HTTP method to include + * @return a copy of the matcher further restricted to only match requests with + * the specified HTTP method + * @since 3.5.0 + */ + public AdditionalPathsEndpointServerWebExchangeMatcher withHttpMethod(HttpMethod httpMethod) { + return new AdditionalPathsEndpointServerWebExchangeMatcher(this.webServerNamespace, this.endpoints, + httpMethod); } @Override @@ -393,7 +426,7 @@ protected ServerWebExchangeMatcher createDelegate(PathMappedEndpoints endpoints) .map(this::getEndpointId) .flatMap((endpointId) -> streamAdditionalPaths(endpoints, endpointId)) .collect(Collectors.toCollection(LinkedHashSet::new)); - List delegateMatchers = getDelegateMatchers(paths); + List delegateMatchers = getDelegateMatchers(paths, this.httpMethod); return (!CollectionUtils.isEmpty(delegateMatchers)) ? new OrServerWebExchangeMatcher(delegateMatchers) : EMPTY_MATCHER; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/AntPathRequestMatcherProvider.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/AntPathRequestMatcherProvider.java new file mode 100644 index 000000000000..430fb4e05fe7 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/AntPathRequestMatcherProvider.java @@ -0,0 +1,45 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.security.servlet; + +import java.util.function.Function; + +import org.springframework.http.HttpMethod; +import org.springframework.security.web.util.matcher.AntPathRequestMatcher; +import org.springframework.security.web.util.matcher.RequestMatcher; + +/** + * {@link RequestMatcherProvider} that provides an {@link AntPathRequestMatcher}. + * + * @author Madhura Bhave + * @author Chris Bono + */ +class AntPathRequestMatcherProvider implements RequestMatcherProvider { + + private final Function pathFactory; + + AntPathRequestMatcherProvider(Function pathFactory) { + this.pathFactory = pathFactory; + } + + @Override + public RequestMatcher getRequestMatcher(String pattern, HttpMethod httpMethod) { + String path = this.pathFactory.apply(pattern); + return new AntPathRequestMatcher(path, (httpMethod != null) ? httpMethod.name() : null); + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/EndpointRequest.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/EndpointRequest.java index 6fcc1630625f..295f3d76f512 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/EndpointRequest.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/EndpointRequest.java @@ -36,12 +36,12 @@ import org.springframework.boot.actuate.endpoint.annotation.Endpoint; import org.springframework.boot.actuate.endpoint.web.PathMappedEndpoints; import org.springframework.boot.actuate.endpoint.web.WebServerNamespace; -import org.springframework.boot.autoconfigure.security.servlet.RequestMatcherProvider; import org.springframework.boot.security.servlet.ApplicationContextRequestMatcher; import org.springframework.boot.web.context.WebServerApplicationContext; import org.springframework.context.ApplicationContext; import org.springframework.core.annotation.MergedAnnotation; import org.springframework.core.annotation.MergedAnnotations; +import org.springframework.http.HttpMethod; import org.springframework.security.web.util.matcher.AntPathRequestMatcher; import org.springframework.security.web.util.matcher.OrRequestMatcher; import org.springframework.security.web.util.matcher.RequestMatcher; @@ -56,6 +56,7 @@ * * @author Madhura Bhave * @author Phillip Webb + * @author Chris Bono * @since 2.0.0 */ public final class EndpointRequest { @@ -217,37 +218,53 @@ protected abstract RequestMatcher createDelegate(WebApplicationContext context, RequestMatcherFactory requestMatcherFactory); protected final List getDelegateMatchers(RequestMatcherFactory requestMatcherFactory, - RequestMatcherProvider matcherProvider, Set paths) { + RequestMatcherProvider matcherProvider, Set paths, HttpMethod httpMethod) { return paths.stream() - .map((path) -> requestMatcherFactory.antPath(matcherProvider, path, "/**")) + .map((path) -> requestMatcherFactory.antPath(matcherProvider, httpMethod, path, "/**")) .collect(Collectors.toCollection(ArrayList::new)); } protected List getLinksMatchers(RequestMatcherFactory requestMatcherFactory, RequestMatcherProvider matcherProvider, String basePath) { List linksMatchers = new ArrayList<>(); - linksMatchers.add(requestMatcherFactory.antPath(matcherProvider, basePath)); - linksMatchers.add(requestMatcherFactory.antPath(matcherProvider, basePath, "/")); + linksMatchers.add(requestMatcherFactory.antPath(matcherProvider, null, basePath)); + linksMatchers.add(requestMatcherFactory.antPath(matcherProvider, null, basePath, "/")); return linksMatchers; } protected RequestMatcherProvider getRequestMatcherProvider(WebApplicationContext context) { + try { + return getRequestMatcherProviderBean(context); + } + catch (NoSuchBeanDefinitionException ex) { + return (pattern, method) -> new AntPathRequestMatcher(pattern, (method != null) ? method.name() : null); + } + } + + private RequestMatcherProvider getRequestMatcherProviderBean(WebApplicationContext context) { try { return context.getBean(RequestMatcherProvider.class); } catch (NoSuchBeanDefinitionException ex) { - return AntPathRequestMatcher::new; + return getAndAdaptDeprecatedRequestMatcherProviderBean(context); } } - protected final String toString(List endpoints, String emptyValue) { + @SuppressWarnings("removal") + private RequestMatcherProvider getAndAdaptDeprecatedRequestMatcherProviderBean(WebApplicationContext context) { + org.springframework.boot.autoconfigure.security.servlet.RequestMatcherProvider bean = context + .getBean(org.springframework.boot.autoconfigure.security.servlet.RequestMatcherProvider.class); + return (pattern, method) -> bean.getRequestMatcher(pattern); + } + + protected String toString(List endpoints, String emptyValue) { return (!endpoints.isEmpty()) ? endpoints.stream() .map(this::getEndpointId) .map(Object::toString) .collect(Collectors.joining(", ", "[", "]")) : emptyValue; } - protected final EndpointId getEndpointId(Object source) { + protected EndpointId getEndpointId(Object source) { if (source instanceof EndpointId endpointId) { return endpointId; } @@ -279,38 +296,53 @@ public static final class EndpointRequestMatcher extends AbstractRequestMatcher private final boolean includeLinks; + private final HttpMethod httpMethod; + private EndpointRequestMatcher(boolean includeLinks) { - this(Collections.emptyList(), Collections.emptyList(), includeLinks); + this(Collections.emptyList(), Collections.emptyList(), includeLinks, null); } private EndpointRequestMatcher(Class[] endpoints, boolean includeLinks) { - this(Arrays.asList((Object[]) endpoints), Collections.emptyList(), includeLinks); + this(Arrays.asList((Object[]) endpoints), Collections.emptyList(), includeLinks, null); } private EndpointRequestMatcher(String[] endpoints, boolean includeLinks) { - this(Arrays.asList((Object[]) endpoints), Collections.emptyList(), includeLinks); + this(Arrays.asList((Object[]) endpoints), Collections.emptyList(), includeLinks, null); } - private EndpointRequestMatcher(List includes, List excludes, boolean includeLinks) { + private EndpointRequestMatcher(List includes, List excludes, boolean includeLinks, + HttpMethod httpMethod) { this.includes = includes; this.excludes = excludes; this.includeLinks = includeLinks; + this.httpMethod = httpMethod; } public EndpointRequestMatcher excluding(Class... endpoints) { List excludes = new ArrayList<>(this.excludes); excludes.addAll(Arrays.asList((Object[]) endpoints)); - return new EndpointRequestMatcher(this.includes, excludes, this.includeLinks); + return new EndpointRequestMatcher(this.includes, excludes, this.includeLinks, null); } public EndpointRequestMatcher excluding(String... endpoints) { List excludes = new ArrayList<>(this.excludes); excludes.addAll(Arrays.asList((Object[]) endpoints)); - return new EndpointRequestMatcher(this.includes, excludes, this.includeLinks); + return new EndpointRequestMatcher(this.includes, excludes, this.includeLinks, null); } public EndpointRequestMatcher excludingLinks() { - return new EndpointRequestMatcher(this.includes, this.excludes, false); + return new EndpointRequestMatcher(this.includes, this.excludes, false, null); + } + + /** + * Restricts the matcher to only consider requests with a particular HTTP method. + * @param httpMethod the HTTP method to include + * @return a copy of the matcher further restricted to only match requests with + * the specified HTTP method + * @since 3.5.0 + */ + public EndpointRequestMatcher withHttpMethod(HttpMethod httpMethod) { + return new EndpointRequestMatcher(this.includes, this.excludes, this.includeLinks, httpMethod); } @Override @@ -324,7 +356,8 @@ protected RequestMatcher createDelegate(WebApplicationContext context, } streamPaths(this.includes, endpoints).forEach(paths::add); streamPaths(this.excludes, endpoints).forEach(paths::remove); - List delegateMatchers = getDelegateMatchers(requestMatcherFactory, matcherProvider, paths); + List delegateMatchers = getDelegateMatchers(requestMatcherFactory, matcherProvider, paths, + this.httpMethod); String basePath = endpoints.getBasePath(); if (this.includeLinks && StringUtils.hasText(basePath)) { delegateMatchers.addAll(getLinksMatchers(requestMatcherFactory, matcherProvider, basePath)); @@ -378,20 +411,35 @@ public static class AdditionalPathsEndpointRequestMatcher extends AbstractReques private final List endpoints; + private final HttpMethod httpMethod; + AdditionalPathsEndpointRequestMatcher(WebServerNamespace webServerNamespace, String... endpoints) { - this(webServerNamespace, Arrays.asList((Object[]) endpoints)); + this(webServerNamespace, Arrays.asList((Object[]) endpoints), null); } AdditionalPathsEndpointRequestMatcher(WebServerNamespace webServerNamespace, Class... endpoints) { - this(webServerNamespace, Arrays.asList((Object[]) endpoints)); + this(webServerNamespace, Arrays.asList((Object[]) endpoints), null); } - private AdditionalPathsEndpointRequestMatcher(WebServerNamespace webServerNamespace, List endpoints) { + private AdditionalPathsEndpointRequestMatcher(WebServerNamespace webServerNamespace, List endpoints, + HttpMethod httpMethod) { Assert.notNull(webServerNamespace, "'webServerNamespace' must not be null"); Assert.notNull(endpoints, "'endpoints' must not be null"); Assert.notEmpty(endpoints, "'endpoints' must not be empty"); this.webServerNamespace = webServerNamespace; this.endpoints = endpoints; + this.httpMethod = httpMethod; + } + + /** + * Restricts the matcher to only consider requests with a particular HTTP method. + * @param httpMethod the HTTP method to include + * @return a copy of the matcher further restricted to only match requests with + * the specified HTTP method + * @since 3.5.0 + */ + public AdditionalPathsEndpointRequestMatcher withHttpMethod(HttpMethod httpMethod) { + return new AdditionalPathsEndpointRequestMatcher(this.webServerNamespace, this.endpoints, httpMethod); } @Override @@ -410,7 +458,8 @@ protected RequestMatcher createDelegate(WebApplicationContext context, .map(this::getEndpointId) .flatMap((endpointId) -> streamAdditionalPaths(endpoints, endpointId)) .collect(Collectors.toCollection(LinkedHashSet::new)); - List delegateMatchers = getDelegateMatchers(requestMatcherFactory, matcherProvider, paths); + List delegateMatchers = getDelegateMatchers(requestMatcherFactory, matcherProvider, paths, + this.httpMethod); return (!CollectionUtils.isEmpty(delegateMatchers)) ? new OrRequestMatcher(delegateMatchers) : EMPTY_MATCHER; } @@ -432,12 +481,12 @@ public String toString() { */ private static final class RequestMatcherFactory { - RequestMatcher antPath(RequestMatcherProvider matcherProvider, String... parts) { + RequestMatcher antPath(RequestMatcherProvider matcherProvider, HttpMethod httpMethod, String... parts) { StringBuilder pattern = new StringBuilder(); for (String part : parts) { pattern.append(part); } - return matcherProvider.getRequestMatcher(pattern.toString()); + return matcherProvider.getRequestMatcher(pattern.toString(), httpMethod); } } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/RequestMatcherProvider.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/RequestMatcherProvider.java new file mode 100644 index 000000000000..f68ffe918d71 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/RequestMatcherProvider.java @@ -0,0 +1,42 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.security.servlet; + +import org.springframework.http.HttpMethod; +import org.springframework.security.web.util.matcher.RequestMatcher; + +/** + * Interface that can be used to provide a {@link RequestMatcher} that can be used with + * Spring Security. + * + * @author Madhura Bhave + * @author Chris Bono + * @since 3.5.0 + */ +@FunctionalInterface +public interface RequestMatcherProvider { + + /** + * Return the {@link RequestMatcher} to be used for the specified pattern and http + * method. + * @param pattern the request pattern + * @param httpMethod the http method + * @return a request matcher + */ + RequestMatcher getRequestMatcher(String pattern, HttpMethod httpMethod); + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/SecurityRequestMatchersManagementContextConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/SecurityRequestMatchersManagementContextConfiguration.java index af67a5257002..260620f34fa3 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/SecurityRequestMatchersManagementContextConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/security/servlet/SecurityRequestMatchersManagementContextConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,8 +24,6 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; -import org.springframework.boot.autoconfigure.security.servlet.AntPathRequestMatcherProvider; -import org.springframework.boot.autoconfigure.security.servlet.RequestMatcherProvider; import org.springframework.boot.autoconfigure.web.servlet.DispatcherServletPath; import org.springframework.boot.autoconfigure.web.servlet.JerseyApplicationPath; import org.springframework.context.annotation.Bean; diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslHealthIndicatorProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslHealthIndicatorProperties.java index eb897d70eb09..f88e7d33b97d 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslHealthIndicatorProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslHealthIndicatorProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ * @author Jonatan Ivanov * @since 3.4.0 */ -@ConfigurationProperties(prefix = "management.health.ssl") +@ConfigurationProperties("management.health.ssl") public class SslHealthIndicatorProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslMeterBinder.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslMeterBinder.java new file mode 100644 index 000000000000..9e9c93a2cf62 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslMeterBinder.java @@ -0,0 +1,272 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.ssl; + +import java.time.Clock; +import java.time.Duration; +import java.time.Instant; +import java.time.temporal.ChronoUnit; +import java.util.ArrayList; +import java.util.Collection; +import java.util.Comparator; +import java.util.EnumSet; +import java.util.HashSet; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; + +import io.micrometer.core.instrument.Gauge; +import io.micrometer.core.instrument.MeterRegistry; +import io.micrometer.core.instrument.MultiGauge; +import io.micrometer.core.instrument.MultiGauge.Row; +import io.micrometer.core.instrument.Tags; +import io.micrometer.core.instrument.TimeGauge; +import io.micrometer.core.instrument.binder.MeterBinder; + +import org.springframework.boot.info.SslInfo; +import org.springframework.boot.info.SslInfo.BundleInfo; +import org.springframework.boot.info.SslInfo.CertificateChainInfo; +import org.springframework.boot.info.SslInfo.CertificateInfo; +import org.springframework.boot.info.SslInfo.CertificateValidityInfo; +import org.springframework.boot.info.SslInfo.CertificateValidityInfo.Status; +import org.springframework.boot.ssl.SslBundles; + +/** + * {@link MeterBinder} which registers the SSL chain validity (soonest to expire + * certificate in the chain) as a {@link TimeGauge}. Also contributes two {@link Gauge + * gauges} to count the valid and invalid chains. + * + * @author Moritz Halbritter + */ +class SslMeterBinder implements MeterBinder { + + private static final String CHAINS_METRIC_NAME = "ssl.chains"; + + private static final String CHAIN_EXPIRY_METRIC_NAME = "ssl.chain.expiry"; + + private final Clock clock; + + private final SslInfo sslInfo; + + private final BundleMetrics bundleMetrics = new BundleMetrics(); + + SslMeterBinder(SslInfo sslInfo, SslBundles sslBundles) { + this(sslInfo, sslBundles, Clock.systemDefaultZone()); + } + + SslMeterBinder(SslInfo sslInfo, SslBundles sslBundles, Clock clock) { + this.clock = clock; + this.sslInfo = sslInfo; + sslBundles.addBundleRegisterHandler((bundleName, ignored) -> onBundleChange(bundleName)); + for (String bundleName : sslBundles.getBundleNames()) { + sslBundles.addBundleUpdateHandler(bundleName, (ignored) -> onBundleChange(bundleName)); + } + } + + private void onBundleChange(String bundleName) { + BundleInfo bundle = this.sslInfo.getBundle(bundleName); + this.bundleMetrics.updateBundle(bundle); + for (MeterRegistry meterRegistry : this.bundleMetrics.getMeterRegistries()) { + createOrUpdateBundleMetrics(meterRegistry, bundle); + } + } + + @Override + public void bindTo(MeterRegistry meterRegistry) { + for (BundleInfo bundle : this.sslInfo.getBundles()) { + createOrUpdateBundleMetrics(meterRegistry, bundle); + } + Gauge.builder(CHAINS_METRIC_NAME, () -> countChainsByStatus(Status.VALID)) + .tag("status", "valid") + .register(meterRegistry); + Gauge.builder(CHAINS_METRIC_NAME, () -> countChainsByStatus(Status.EXPIRED)) + .tag("status", "expired") + .register(meterRegistry); + Gauge.builder(CHAINS_METRIC_NAME, () -> countChainsByStatus(Status.NOT_YET_VALID)) + .tag("status", "not-yet-valid") + .register(meterRegistry); + Gauge.builder(CHAINS_METRIC_NAME, () -> countChainsByStatus(Status.WILL_EXPIRE_SOON)) + .tag("status", "will-expire-soon") + .register(meterRegistry); + } + + private void createOrUpdateBundleMetrics(MeterRegistry meterRegistry, BundleInfo bundle) { + MultiGauge multiGauge = this.bundleMetrics.getGauge(bundle, meterRegistry); + List> rows = new ArrayList<>(); + for (CertificateChainInfo chain : bundle.getCertificateChains()) { + Row row = createRowForChain(bundle, chain); + if (row != null) { + rows.add(row); + } + } + multiGauge.register(rows, true); + } + + private Row createRowForChain(BundleInfo bundle, CertificateChainInfo chain) { + CertificateInfo leastValidCertificate = chain.getCertificates() + .stream() + .min(Comparator.comparing(CertificateInfo::getValidityEnds)) + .orElse(null); + if (leastValidCertificate == null) { + return null; + } + Tags tags = Tags.of("chain", chain.getAlias(), "bundle", bundle.getName(), "certificate", + leastValidCertificate.getSerialNumber()); + return Row.of(tags, leastValidCertificate, this::getChainExpiry); + } + + private long countChainsByStatus(Status status) { + long count = 0; + for (BundleInfo bundle : this.bundleMetrics.getBundles()) { + for (CertificateChainInfo chain : bundle.getCertificateChains()) { + if (getChainStatus(chain) == status) { + count++; + } + } + } + return count; + } + + private Status getChainStatus(CertificateChainInfo chain) { + EnumSet statuses = EnumSet.noneOf(Status.class); + for (CertificateInfo certificate : chain.getCertificates()) { + CertificateValidityInfo validity = certificate.getValidity(); + statuses.add(validity.getStatus()); + } + if (statuses.contains(Status.EXPIRED)) { + return Status.EXPIRED; + } + if (statuses.contains(Status.NOT_YET_VALID)) { + return Status.NOT_YET_VALID; + } + if (statuses.contains(Status.WILL_EXPIRE_SOON)) { + return Status.WILL_EXPIRE_SOON; + } + return statuses.isEmpty() ? null : Status.VALID; + } + + private long getChainExpiry(CertificateInfo certificate) { + Duration valid = Duration.between(Instant.now(this.clock), certificate.getValidityEnds()); + return valid.get(ChronoUnit.SECONDS); + } + + /** + * Manages bundles and their metrics. + */ + private static final class BundleMetrics { + + private final Map gauges = new ConcurrentHashMap<>(); + + /** + * Gets (or creates) a {@link MultiGauge} for the given bundle and meter registry. + * @param bundleInfo the bundle + * @param meterRegistry the meter registry + * @return the {@link MultiGauge} + */ + MultiGauge getGauge(BundleInfo bundleInfo, MeterRegistry meterRegistry) { + Gauges gauges = this.gauges.computeIfAbsent(bundleInfo.getName(), + (ignored) -> Gauges.emptyGauges(bundleInfo)); + return gauges.getGauge(meterRegistry); + } + + /** + * Returns all bundles. + * @return all bundles + */ + Collection getBundles() { + List result = new ArrayList<>(); + for (Gauges metrics : this.gauges.values()) { + result.add(metrics.bundle()); + } + return result; + } + + /** + * Returns all meter registries. + * @return all meter registries + */ + Collection getMeterRegistries() { + Set result = new HashSet<>(); + for (Gauges metrics : this.gauges.values()) { + result.addAll(metrics.getMeterRegistries()); + } + return result; + } + + /** + * Updates the given bundle. + * @param bundle the updated bundle + */ + void updateBundle(BundleInfo bundle) { + this.gauges.computeIfPresent(bundle.getName(), (key, oldValue) -> oldValue.withBundle(bundle)); + } + + /** + * Manages the {@link MultiGauge MultiGauges} associated to a bundle. + * + * @param bundle the bundle + * @param multiGauges mapping from meter registry to {@link MultiGauge} + */ + private record Gauges(BundleInfo bundle, Map multiGauges) { + + /** + * Gets (or creates) the {@link MultiGauge} for the given meter registry. + * @param meterRegistry the meter registry + * @return the {@link MultiGauge} + */ + MultiGauge getGauge(MeterRegistry meterRegistry) { + return this.multiGauges.computeIfAbsent(meterRegistry, (ignored) -> createGauge(meterRegistry)); + } + + /** + * Returns a copy of this bundle with an updated {@link BundleInfo}. + * @param bundle the updated {@link BundleInfo} + * @return the copy of this bundle with an updated {@link BundleInfo} + */ + Gauges withBundle(BundleInfo bundle) { + return new Gauges(bundle, this.multiGauges); + } + + /** + * Returns all meter registries. + * @return all meter registries + */ + Set getMeterRegistries() { + return this.multiGauges.keySet(); + } + + private MultiGauge createGauge(MeterRegistry meterRegistry) { + return MultiGauge.builder(CHAIN_EXPIRY_METRIC_NAME) + .baseUnit("seconds") + .description("SSL chain expiry") + .register(meterRegistry); + } + + /** + * Creates an instance with an empty gauge mapping. + * @param bundle the {@link BundleInfo} associated with the new instance + * @return the new instance + */ + static Gauges emptyGauges(BundleInfo bundle) { + return new Gauges(bundle, new ConcurrentHashMap<>()); + } + } + + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslObservabilityAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslObservabilityAutoConfiguration.java new file mode 100644 index 000000000000..0a074d0a99c2 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/ssl/SslObservabilityAutoConfiguration.java @@ -0,0 +1,58 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.ssl; + +import io.micrometer.core.instrument.MeterRegistry; + +import org.springframework.boot.actuate.autoconfigure.metrics.CompositeMeterRegistryAutoConfiguration; +import org.springframework.boot.actuate.autoconfigure.metrics.MetricsAutoConfiguration; +import org.springframework.boot.autoconfigure.AutoConfiguration; +import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; +import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.boot.autoconfigure.ssl.SslAutoConfiguration; +import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.boot.info.SslInfo; +import org.springframework.boot.ssl.SslBundles; +import org.springframework.context.annotation.Bean; + +/** + * {@link EnableAutoConfiguration Auto-configuration} for SSL observability. + * + * @author Moritz Halbritter + * @since 3.5.0 + */ +@AutoConfiguration(after = { MetricsAutoConfiguration.class, CompositeMeterRegistryAutoConfiguration.class, + SslAutoConfiguration.class }) +@ConditionalOnClass(MeterRegistry.class) +@ConditionalOnBean({ MeterRegistry.class, SslBundles.class }) +@EnableConfigurationProperties(SslHealthIndicatorProperties.class) +public class SslObservabilityAutoConfiguration { + + @Bean + SslMeterBinder sslMeterBinder(SslInfo sslInfo, SslBundles sslBundles) { + return new SslMeterBinder(sslInfo, sslBundles); + } + + @Bean + @ConditionalOnMissingBean + SslInfo sslInfoProvider(SslBundles sslBundles, SslHealthIndicatorProperties properties) { + return new SslInfo(sslBundles, properties.getCertificateValidityWarningThreshold()); + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/system/DiskSpaceHealthIndicatorProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/system/DiskSpaceHealthIndicatorProperties.java index f2b625463ad2..a7e7dfaa0a9c 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/system/DiskSpaceHealthIndicatorProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/system/DiskSpaceHealthIndicatorProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,7 +30,7 @@ * @author Stephane Nicoll * @since 1.2.0 */ -@ConfigurationProperties(prefix = "management.health.diskspace") +@ConfigurationProperties("management.health.diskspace") public class DiskSpaceHealthIndicatorProperties { /** @@ -56,7 +56,7 @@ public DataSize getThreshold() { } public void setThreshold(DataSize threshold) { - Assert.isTrue(!threshold.isNegative(), "threshold must be greater than or equal to 0"); + Assert.isTrue(!threshold.isNegative(), "'threshold' must be greater than or equal to 0"); this.threshold = threshold; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/BravePropagationConfigurations.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/BravePropagationConfigurations.java index 97bca941de8b..3e1176510bb1 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/BravePropagationConfigurations.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/BravePropagationConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,8 +34,8 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.actuate.autoconfigure.tracing.TracingProperties.Baggage.Correlation; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -52,7 +52,7 @@ class BravePropagationConfigurations { * Propagates traces but no baggage. */ @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(value = "management.tracing.baggage.enabled", havingValue = "false") + @ConditionalOnBooleanProperty(name = "management.tracing.baggage.enabled", havingValue = false) static class PropagationWithoutBaggage { @Bean @@ -68,7 +68,7 @@ CompositePropagationFactory propagationFactory(TracingProperties properties) { * Propagates traces and baggage. */ @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(value = "management.tracing.baggage.enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "management.tracing.baggage.enabled", matchIfMissing = true) @EnableConfigurationProperties(TracingProperties.class) static class PropagationWithBaggage { @@ -142,8 +142,7 @@ CorrelationScopeDecorator.Builder mdcCorrelationScopeDecoratorBuilder( @Bean @Order(0) - @ConditionalOnProperty(prefix = "management.tracing.baggage.correlation", name = "enabled", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "management.tracing.baggage.correlation.enabled", matchIfMissing = true) CorrelationScopeCustomizer correlationFieldsCorrelationScopeCustomizer() { return (builder) -> { Correlation correlationProperties = this.tracingProperties.getBaggage().getCorrelation(); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/LocalBaggageFields.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/LocalBaggageFields.java index dfd3f495ae7c..aaf3311d6969 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/LocalBaggageFields.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/LocalBaggageFields.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ class LocalBaggageFields { private final List fields; LocalBaggageFields(List fields) { - Assert.notNull(fields, "fields must not be null"); + Assert.notNull(fields, "'fields' must not be null"); this.fields = fields; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/MicrometerTracingAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/MicrometerTracingAutoConfiguration.java index ce31cd4d5b88..748f1f45bb80 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/MicrometerTracingAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/MicrometerTracingAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,13 +33,11 @@ import org.springframework.beans.factory.BeanFactory; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; -import org.springframework.boot.autoconfigure.condition.AnyNestedCondition; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Conditional; import org.springframework.context.annotation.Configuration; import org.springframework.core.Ordered; import org.springframework.core.annotation.Order; @@ -104,7 +102,7 @@ public PropagatingReceiverTracingObservationHandler propagatingReceiverTracin @Configuration(proxyBeanMethods = false) @ConditionalOnClass(Advice.class) - @Conditional(ObservationAnnotationsEnabledCondition.class) + @ConditionalOnBooleanProperty("management.observations.annotations.enabled") static class SpanAspectConfiguration { @Bean @@ -152,22 +150,4 @@ public String resolve(String expression, Object parameter) { } - static final class ObservationAnnotationsEnabledCondition extends AnyNestedCondition { - - ObservationAnnotationsEnabledCondition() { - super(ConfigurationPhase.PARSE_CONFIGURATION); - } - - @ConditionalOnProperty(prefix = "micrometer.observations.annotations", name = "enabled", havingValue = "true") - static class MicrometerObservationsEnabledCondition { - - } - - @ConditionalOnProperty(prefix = "management.observations.annotations", name = "enabled", havingValue = "true") - static class ManagementObservationsEnabledCondition { - - } - - } - } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryPropagationConfigurations.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryPropagationConfigurations.java index cb2210ec516e..23064eb56a82 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryPropagationConfigurations.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryPropagationConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,8 +24,8 @@ import io.micrometer.tracing.otel.propagation.BaggageTextMapPropagator; import io.opentelemetry.context.propagation.TextMapPropagator; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -42,7 +42,7 @@ class OpenTelemetryPropagationConfigurations { * Propagates traces but no baggage. */ @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "management.tracing.baggage", name = "enabled", havingValue = "false") + @ConditionalOnBooleanProperty(name = "management.tracing.baggage.enabled", havingValue = false) @EnableConfigurationProperties(TracingProperties.class) static class PropagationWithoutBaggage { @@ -58,7 +58,7 @@ TextMapPropagator textMapPropagator(TracingProperties properties) { * Propagates traces and baggage. */ @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "management.tracing.baggage", name = "enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "management.tracing.baggage.enabled", matchIfMissing = true) @EnableConfigurationProperties(TracingProperties.class) static class PropagationWithBaggage { @@ -80,8 +80,7 @@ TextMapPropagator textMapPropagatorWithBaggage(OtelCurrentTraceContext otelCurre @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(prefix = "management.tracing.baggage.correlation", name = "enabled", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "management.tracing.baggage.correlation.enabled", matchIfMissing = true) Slf4JBaggageEventListener otelSlf4JBaggageEventListener() { return new Slf4JBaggageEventListener(this.tracingProperties.getBaggage().getCorrelation().getFields()); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryTracingAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryTracingAutoConfiguration.java index 3f88a7009ff3..0e5c27ac1914 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryTracingAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryTracingAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -115,12 +115,20 @@ SpanProcessors spanProcessors(ObjectProvider spanProcessors) { } @Bean + @ConditionalOnMissingBean BatchSpanProcessor otelSpanProcessor(SpanExporters spanExporters, ObjectProvider spanExportingPredicates, ObjectProvider spanReporters, ObjectProvider spanFilters, ObjectProvider meterProvider) { - BatchSpanProcessorBuilder builder = BatchSpanProcessor - .builder(new CompositeSpanExporter(spanExporters.list(), spanExportingPredicates.orderedStream().toList(), - spanReporters.orderedStream().toList(), spanFilters.orderedStream().toList())); + TracingProperties.OpenTelemetry.Export properties = this.tracingProperties.getOpentelemetry().getExport(); + CompositeSpanExporter spanExporter = new CompositeSpanExporter(spanExporters.list(), + spanExportingPredicates.orderedStream().toList(), spanReporters.orderedStream().toList(), + spanFilters.orderedStream().toList()); + BatchSpanProcessorBuilder builder = BatchSpanProcessor.builder(spanExporter) + .setExportUnsampledSpans(properties.isIncludeUnsampled()) + .setExporterTimeout(properties.getTimeout()) + .setMaxExportBatchSize(properties.getMaxBatchSize()) + .setMaxQueueSize(properties.getMaxQueueSize()) + .setScheduleDelay(properties.getScheduleDelay()); meterProvider.ifAvailable(builder::setMeterProvider); return builder.build(); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/SpanExporters.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/SpanExporters.java index a44f8ce0e035..1b7a5b18adcf 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/SpanExporters.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/SpanExporters.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -68,7 +68,7 @@ static SpanExporters of(SpanExporter... spanExporters) { * @return the constructed {@link SpanExporters} instance */ static SpanExporters of(Collection spanExporters) { - Assert.notNull(spanExporters, "SpanExporters must not be null"); + Assert.notNull(spanExporters, "'spanExporters' must not be null"); List copy = List.copyOf(spanExporters); return () -> copy; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/SpanProcessors.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/SpanProcessors.java index ca8c55498d07..30cda9f7f741 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/SpanProcessors.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/SpanProcessors.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -68,7 +68,7 @@ static SpanProcessors of(SpanProcessor... spanProcessors) { * @return the constructed {@link SpanProcessors} instance */ static SpanProcessors of(Collection spanProcessors) { - Assert.notNull(spanProcessors, "SpanProcessors must not be null"); + Assert.notNull(spanProcessors, "'spanProcessors' must not be null"); List copy = List.copyOf(spanProcessors); return () -> copy; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/TracingProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/TracingProperties.java index 175ff03a8175..7fe030ca1655 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/TracingProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/TracingProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,6 +16,7 @@ package org.springframework.boot.actuate.autoconfigure.tracing; +import java.time.Duration; import java.util.ArrayList; import java.util.List; @@ -51,6 +52,11 @@ public class TracingProperties { */ private final Brave brave = new Brave(); + /** + * OpenTelemetry configuration. + */ + private final OpenTelemetry opentelemetry = new OpenTelemetry(); + public Sampling getSampling() { return this.sampling; } @@ -67,6 +73,10 @@ public Brave getBrave() { return this.brave; } + public OpenTelemetry getOpentelemetry() { + return this.opentelemetry; + } + public static class Sampling { /** @@ -293,4 +303,88 @@ public void setSpanJoiningSupported(boolean spanJoiningSupported) { } + public static class OpenTelemetry { + + /** + * Span export configuration. + */ + private final Export export = new Export(); + + public Export getExport() { + return this.export; + } + + public static class Export { + + /** + * Whether unsampled spans should be exported. + */ + private boolean includeUnsampled; + + /** + * Maximum time an export will be allowed to run before being cancelled. + */ + private Duration timeout = Duration.ofSeconds(30); + + /** + * Maximum batch size for each export. This must be less than or equal to + * 'maxQueueSize'. + */ + private int maxBatchSize = 512; + + /** + * Maximum number of spans that are kept in the queue before they will be + * dropped. + */ + private int maxQueueSize = 2048; + + /** + * The delay interval between two consecutive exports. + */ + private Duration scheduleDelay = Duration.ofSeconds(5); + + public boolean isIncludeUnsampled() { + return this.includeUnsampled; + } + + public void setIncludeUnsampled(boolean includeUnsampled) { + this.includeUnsampled = includeUnsampled; + } + + public Duration getTimeout() { + return this.timeout; + } + + public void setTimeout(Duration timeout) { + this.timeout = timeout; + } + + public int getMaxBatchSize() { + return this.maxBatchSize; + } + + public void setMaxBatchSize(int maxBatchSize) { + this.maxBatchSize = maxBatchSize; + } + + public int getMaxQueueSize() { + return this.maxQueueSize; + } + + public void setMaxQueueSize(int maxQueueSize) { + this.maxQueueSize = maxQueueSize; + } + + public Duration getScheduleDelay() { + return this.scheduleDelay; + } + + public void setScheduleDelay(Duration scheduleDelay) { + this.scheduleDelay = scheduleDelay; + } + + } + + } + } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpGrpcSpanExporterBuilderCustomizer.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpGrpcSpanExporterBuilderCustomizer.java new file mode 100644 index 000000000000..3a82e97f6746 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpGrpcSpanExporterBuilderCustomizer.java @@ -0,0 +1,37 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.tracing.otlp; + +import io.opentelemetry.exporter.otlp.trace.OtlpGrpcSpanExporterBuilder; + +/** + * Callback interface that can be implemented by beans wishing to customize the + * {@link OtlpGrpcSpanExporterBuilder} whilst retaining default auto-configuration. + * + * @author Dmytro Nosan + * @since 3.5.0 + */ +@FunctionalInterface +public interface OtlpGrpcSpanExporterBuilderCustomizer { + + /** + * Customize the {@link OtlpGrpcSpanExporterBuilder}. + * @param builder the builder to customize + */ + void customize(OtlpGrpcSpanExporterBuilder builder); + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpHttpSpanExporterBuilderCustomizer.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpHttpSpanExporterBuilderCustomizer.java new file mode 100644 index 000000000000..45ef2b3ac902 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpHttpSpanExporterBuilderCustomizer.java @@ -0,0 +1,37 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.tracing.otlp; + +import io.opentelemetry.exporter.otlp.http.trace.OtlpHttpSpanExporterBuilder; + +/** + * Callback interface that can be implemented by beans wishing to customize the + * {@link OtlpHttpSpanExporterBuilder} whilst retaining default auto-configuration. + * + * @author Dmytro Nosan + * @since 3.5.0 + */ +@FunctionalInterface +public interface OtlpHttpSpanExporterBuilderCustomizer { + + /** + * Customize the {@link OtlpHttpSpanExporterBuilder}. + * @param builder the builder to customize + */ + void customize(OtlpHttpSpanExporterBuilder builder); + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingConfigurations.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingConfigurations.java index 6f0494379b8a..deeaddecd2fa 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingConfigurations.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,11 +18,13 @@ import java.util.Locale; +import io.opentelemetry.api.metrics.MeterProvider; import io.opentelemetry.exporter.otlp.http.trace.OtlpHttpSpanExporter; import io.opentelemetry.exporter.otlp.http.trace.OtlpHttpSpanExporterBuilder; import io.opentelemetry.exporter.otlp.trace.OtlpGrpcSpanExporter; import io.opentelemetry.exporter.otlp.trace.OtlpGrpcSpanExporterBuilder; +import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.actuate.autoconfigure.tracing.ConditionalOnEnabledTracing; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; @@ -44,7 +46,7 @@ static class ConnectionDetails { @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(prefix = "management.otlp.tracing", name = "endpoint") + @ConditionalOnProperty("management.otlp.tracing.endpoint") OtlpTracingConnectionDetails otlpTracingConnectionDetails(OtlpTracingProperties properties) { return new PropertiesOtlpTracingConnectionDetails(properties); } @@ -79,29 +81,34 @@ public String getUrl(Transport transport) { static class Exporters { @Bean - @ConditionalOnProperty(prefix = "management.otlp.tracing", name = "transport", havingValue = "http", - matchIfMissing = true) + @ConditionalOnProperty(name = "management.otlp.tracing.transport", havingValue = "http", matchIfMissing = true) OtlpHttpSpanExporter otlpHttpSpanExporter(OtlpTracingProperties properties, - OtlpTracingConnectionDetails connectionDetails) { + OtlpTracingConnectionDetails connectionDetails, ObjectProvider meterProvider, + ObjectProvider customizers) { OtlpHttpSpanExporterBuilder builder = OtlpHttpSpanExporter.builder() .setEndpoint(connectionDetails.getUrl(Transport.HTTP)) .setTimeout(properties.getTimeout()) .setConnectTimeout(properties.getConnectTimeout()) .setCompression(properties.getCompression().name().toLowerCase(Locale.ROOT)); properties.getHeaders().forEach(builder::addHeader); + meterProvider.ifAvailable(builder::setMeterProvider); + customizers.orderedStream().forEach((customizer) -> customizer.customize(builder)); return builder.build(); } @Bean - @ConditionalOnProperty(prefix = "management.otlp.tracing", name = "transport", havingValue = "grpc") + @ConditionalOnProperty(name = "management.otlp.tracing.transport", havingValue = "grpc") OtlpGrpcSpanExporter otlpGrpcSpanExporter(OtlpTracingProperties properties, - OtlpTracingConnectionDetails connectionDetails) { + OtlpTracingConnectionDetails connectionDetails, ObjectProvider meterProvider, + ObjectProvider customizers) { OtlpGrpcSpanExporterBuilder builder = OtlpGrpcSpanExporter.builder() .setEndpoint(connectionDetails.getUrl(Transport.GRPC)) .setTimeout(properties.getTimeout()) .setConnectTimeout(properties.getConnectTimeout()) .setCompression(properties.getCompression().name().toLowerCase(Locale.ROOT)); properties.getHeaders().forEach(builder::addHeader); + meterProvider.ifAvailable(builder::setMeterProvider); + customizers.orderedStream().forEach((customizer) -> customizer.customize(builder)); return builder.build(); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingConnectionDetails.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingConnectionDetails.java index bb50e4e7bc86..ac0efaed9fd8 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingConnectionDetails.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingConnectionDetails.java @@ -30,7 +30,7 @@ public interface OtlpTracingConnectionDetails extends ConnectionDetails { /** * Address to where tracing will be published. * @return the address to where tracing will be published - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of {@link #getUrl(Transport)} + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of {@link #getUrl(Transport)} */ @Deprecated(since = "3.4.0", forRemoval = true) default String getUrl() { diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/prometheus/PrometheusSimpleclientExemplarsAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/prometheus/PrometheusSimpleclientExemplarsAutoConfiguration.java deleted file mode 100644 index 9dcc65aa831e..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/prometheus/PrometheusSimpleclientExemplarsAutoConfiguration.java +++ /dev/null @@ -1,97 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.tracing.prometheus; - -import io.micrometer.tracing.Span; -import io.micrometer.tracing.Tracer; -import io.prometheus.client.exemplars.tracer.common.SpanContextSupplier; - -import org.springframework.beans.factory.ObjectProvider; -import org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus.PrometheusSimpleclientMetricsExportAutoConfiguration; -import org.springframework.boot.actuate.autoconfigure.tracing.MicrometerTracingAutoConfiguration; -import org.springframework.boot.autoconfigure.AutoConfiguration; -import org.springframework.boot.autoconfigure.EnableAutoConfiguration; -import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.context.annotation.Bean; -import org.springframework.util.function.SingletonSupplier; - -/** - * {@link EnableAutoConfiguration Auto-configuration} for Prometheus Exemplars with - * Micrometer Tracing. - * - * @author Jonatan Ivanov - * @since 3.0.0 - * @deprecated since 3.3.0 for removal in 3.5.0 - */ -@SuppressWarnings("removal") -@Deprecated(forRemoval = true, since = "3.3.0") -@AutoConfiguration(before = PrometheusSimpleclientMetricsExportAutoConfiguration.class, - after = MicrometerTracingAutoConfiguration.class) -@ConditionalOnBean(Tracer.class) -@ConditionalOnClass({ Tracer.class, SpanContextSupplier.class }) -public class PrometheusSimpleclientExemplarsAutoConfiguration { - - @Bean - @ConditionalOnMissingBean - SpanContextSupplier spanContextSupplier(ObjectProvider tracerProvider) { - return new LazyTracingSpanContextSupplier(tracerProvider); - } - - /** - * Since the MeterRegistry can depend on the {@link Tracer} (Exemplars) and the - * {@link Tracer} can depend on the MeterRegistry (recording metrics), this - * {@link SpanContextSupplier} breaks the cycle by lazily loading the {@link Tracer}. - */ - static class LazyTracingSpanContextSupplier implements SpanContextSupplier { - - private final SingletonSupplier tracer; - - LazyTracingSpanContextSupplier(ObjectProvider tracerProvider) { - this.tracer = SingletonSupplier.of(tracerProvider::getObject); - } - - @Override - public String getTraceId() { - Span currentSpan = currentSpan(); - return (currentSpan != null) ? currentSpan.context().traceId() : null; - } - - @Override - public String getSpanId() { - Span currentSpan = currentSpan(); - return (currentSpan != null) ? currentSpan.context().spanId() : null; - } - - @Override - public boolean isSampled() { - Span currentSpan = currentSpan(); - if (currentSpan == null) { - return false; - } - Boolean sampled = currentSpan.context().sampled(); - return sampled != null && sampled; - } - - private Span currentSpan() { - return this.tracer.obtain().currentSpan(); - } - - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinConfigurations.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinConfigurations.java index 76ecc4cd830d..9a95365b6f2f 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinConfigurations.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,6 @@ import zipkin2.reporter.SpanBytesEncoder; import zipkin2.reporter.brave.AsyncZipkinSpanHandler; import zipkin2.reporter.brave.MutableSpanBytesEncoder; -import zipkin2.reporter.urlconnection.URLConnectionSender; import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.actuate.autoconfigure.tracing.ConditionalOnEnabledTracing; @@ -40,115 +39,25 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.context.properties.EnableConfigurationProperties; -import org.springframework.boot.web.client.RestTemplateBuilder; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Import; -import org.springframework.web.client.RestTemplate; -import org.springframework.web.reactive.function.client.WebClient; /** * Configurations for Zipkin. Those are imported by {@link ZipkinAutoConfiguration}. * * @author Moritz Halbritter * @author Stefan Bratanov + * @author Wick Dynex */ class ZipkinConfigurations { @Configuration(proxyBeanMethods = false) - @Import({ UrlConnectionSenderConfiguration.class, WebClientSenderConfiguration.class, - RestTemplateSenderConfiguration.class, HttpClientSenderConfiguration.class }) + @Import({ HttpClientSenderConfiguration.class }) static class SenderConfiguration { } - @Configuration(proxyBeanMethods = false) - @ConditionalOnClass(URLConnectionSender.class) - @EnableConfigurationProperties(ZipkinProperties.class) - static class UrlConnectionSenderConfiguration { - - @Bean - @ConditionalOnMissingBean(BytesMessageSender.class) - URLConnectionSender urlConnectionSender(ZipkinProperties properties, Encoding encoding, - ObjectProvider connectionDetailsProvider, - ObjectProvider endpointSupplierFactoryProvider) { - ZipkinConnectionDetails connectionDetails = connectionDetailsProvider - .getIfAvailable(() -> new PropertiesZipkinConnectionDetails(properties)); - HttpEndpointSupplier.Factory endpointSupplierFactory = endpointSupplierFactoryProvider - .getIfAvailable(HttpEndpointSuppliers::constantFactory); - URLConnectionSender.Builder builder = URLConnectionSender.newBuilder(); - builder.connectTimeout((int) properties.getConnectTimeout().toMillis()); - builder.readTimeout((int) properties.getReadTimeout().toMillis()); - builder.endpointSupplierFactory(endpointSupplierFactory); - builder.endpoint(connectionDetails.getSpanEndpoint()); - builder.encoding(encoding); - return builder.build(); - } - - } - - @Configuration(proxyBeanMethods = false) - @ConditionalOnClass(RestTemplate.class) - @EnableConfigurationProperties(ZipkinProperties.class) - static class RestTemplateSenderConfiguration { - - @Bean - @ConditionalOnMissingBean(BytesMessageSender.class) - @SuppressWarnings({ "deprecation", "removal" }) - ZipkinRestTemplateSender restTemplateSender(ZipkinProperties properties, Encoding encoding, - ObjectProvider customizers, - ObjectProvider connectionDetailsProvider, - ObjectProvider endpointSupplierFactoryProvider) { - ZipkinConnectionDetails connectionDetails = connectionDetailsProvider - .getIfAvailable(() -> new PropertiesZipkinConnectionDetails(properties)); - HttpEndpointSupplier.Factory endpointSupplierFactory = endpointSupplierFactoryProvider - .getIfAvailable(HttpEndpointSuppliers::constantFactory); - RestTemplateBuilder restTemplateBuilder = new RestTemplateBuilder() - .setConnectTimeout(properties.getConnectTimeout()) - .setReadTimeout(properties.getReadTimeout()); - restTemplateBuilder = applyCustomizers(restTemplateBuilder, customizers); - return new ZipkinRestTemplateSender(encoding, endpointSupplierFactory, connectionDetails.getSpanEndpoint(), - restTemplateBuilder.build()); - } - - @SuppressWarnings({ "deprecation", "removal" }) - private RestTemplateBuilder applyCustomizers(RestTemplateBuilder restTemplateBuilder, - ObjectProvider customizers) { - Iterable orderedCustomizers = () -> customizers.orderedStream() - .iterator(); - RestTemplateBuilder currentBuilder = restTemplateBuilder; - for (ZipkinRestTemplateBuilderCustomizer customizer : orderedCustomizers) { - currentBuilder = customizer.customize(currentBuilder); - } - return currentBuilder; - } - - } - - @Configuration(proxyBeanMethods = false) - @ConditionalOnClass(WebClient.class) - @EnableConfigurationProperties(ZipkinProperties.class) - static class WebClientSenderConfiguration { - - @Bean - @ConditionalOnMissingBean(BytesMessageSender.class) - @SuppressWarnings({ "deprecation", "removal" }) - ZipkinWebClientSender webClientSender(ZipkinProperties properties, Encoding encoding, - ObjectProvider customizers, - ObjectProvider connectionDetailsProvider, - ObjectProvider endpointSupplierFactoryProvider) { - ZipkinConnectionDetails connectionDetails = connectionDetailsProvider - .getIfAvailable(() -> new PropertiesZipkinConnectionDetails(properties)); - HttpEndpointSupplier.Factory endpointSupplierFactory = endpointSupplierFactoryProvider - .getIfAvailable(HttpEndpointSuppliers::constantFactory); - WebClient.Builder builder = WebClient.builder(); - customizers.orderedStream().forEach((customizer) -> customizer.customize(builder)); - return new ZipkinWebClientSender(encoding, endpointSupplierFactory, connectionDetails.getSpanEndpoint(), - builder.build(), properties.getConnectTimeout().plus(properties.getReadTimeout())); - } - - } - @Configuration(proxyBeanMethods = false) @ConditionalOnClass(HttpClient.class) @EnableConfigurationProperties(ZipkinProperties.class) diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinRestTemplateBuilderCustomizer.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinRestTemplateBuilderCustomizer.java deleted file mode 100644 index d07600238270..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinRestTemplateBuilderCustomizer.java +++ /dev/null @@ -1,41 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.tracing.zipkin; - -import org.springframework.boot.web.client.RestTemplateBuilder; - -/** - * Callback interface that can be implemented by beans wishing to customize the - * {@link RestTemplateBuilder} used to send spans to Zipkin. - * - * @author Marcin Grzejszczak - * @since 3.0.0 - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of - * {@link ZipkinHttpClientBuilderCustomizer} - */ -@FunctionalInterface -@Deprecated(since = "3.3.0", forRemoval = true) -public interface ZipkinRestTemplateBuilderCustomizer { - - /** - * Customize the rest template builder. - * @param restTemplateBuilder the {@code RestTemplateBuilder} to customize - * @return the customized {@code RestTemplateBuilder} - */ - RestTemplateBuilder customize(RestTemplateBuilder restTemplateBuilder); - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinRestTemplateSender.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinRestTemplateSender.java deleted file mode 100644 index 88bf8d5aec7d..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinRestTemplateSender.java +++ /dev/null @@ -1,53 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.tracing.zipkin; - -import java.net.URI; - -import zipkin2.reporter.Encoding; -import zipkin2.reporter.HttpEndpointSupplier.Factory; - -import org.springframework.http.HttpEntity; -import org.springframework.http.HttpMethod; -import org.springframework.util.MultiValueMap; -import org.springframework.web.client.RestTemplate; - -/** - * An {@link HttpSender} which uses {@link RestTemplate} for HTTP communication. - * - * @author Moritz Halbritter - * @author Stefan Bratanov - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of {@link ZipkinHttpClientSender} - */ -@Deprecated(since = "3.3.0", forRemoval = true) -class ZipkinRestTemplateSender extends HttpSender { - - private final RestTemplate restTemplate; - - ZipkinRestTemplateSender(Encoding encoding, Factory endpointSupplierFactory, String endpoint, - RestTemplate restTemplate) { - super(encoding, endpointSupplierFactory, endpoint); - this.restTemplate = restTemplate; - } - - @Override - void postSpans(URI endpoint, MultiValueMap headers, byte[] body) { - HttpEntity request = new HttpEntity<>(body, headers); - this.restTemplate.exchange(endpoint, HttpMethod.POST, request, Void.class); - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinWebClientBuilderCustomizer.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinWebClientBuilderCustomizer.java deleted file mode 100644 index 206315772140..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinWebClientBuilderCustomizer.java +++ /dev/null @@ -1,41 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.tracing.zipkin; - -import org.springframework.web.reactive.function.client.WebClient; -import org.springframework.web.reactive.function.client.WebClient.Builder; - -/** - * Callback interface that can be implemented by beans wishing to customize the - * {@link Builder} used to send spans to Zipkin. - * - * @author Marcin Grzejszczak - * @since 3.0.0 - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of - * {@link ZipkinHttpClientBuilderCustomizer} - */ -@FunctionalInterface -@Deprecated(since = "3.3.0", forRemoval = true) -public interface ZipkinWebClientBuilderCustomizer { - - /** - * Customize the web client builder. - * @param webClientBuilder the {@code WebClient.Builder} to customize - */ - void customize(WebClient.Builder webClientBuilder); - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinWebClientSender.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinWebClientSender.java deleted file mode 100644 index 8ded275a6dfb..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinWebClientSender.java +++ /dev/null @@ -1,61 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.tracing.zipkin; - -import java.net.URI; -import java.time.Duration; - -import zipkin2.reporter.Encoding; -import zipkin2.reporter.HttpEndpointSupplier.Factory; - -import org.springframework.util.MultiValueMap; -import org.springframework.web.reactive.function.client.WebClient; - -/** - * An {@link HttpSender} which uses {@link WebClient} for HTTP communication. - * - * @author Stefan Bratanov - * @author Moritz Halbritter - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of {@link ZipkinHttpClientSender} - */ -@Deprecated(since = "3.3.0", forRemoval = true) -class ZipkinWebClientSender extends HttpSender { - - private final WebClient webClient; - - private final Duration timeout; - - ZipkinWebClientSender(Encoding encoding, Factory endpointSupplierFactory, String endpoint, WebClient webClient, - Duration timeout) { - super(encoding, endpointSupplierFactory, endpoint); - this.webClient = webClient; - this.timeout = timeout; - } - - @Override - void postSpans(URI endpoint, MultiValueMap headers, byte[] body) { - this.webClient.post() - .uri(endpoint) - .headers((h) -> h.addAll(headers)) - .bodyValue(body) - .retrieve() - .toBodilessEntity() - .timeout(this.timeout) - .block(); - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/wavefront/WavefrontProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/wavefront/WavefrontProperties.java index ce313c3360d1..436a8e6086cf 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/wavefront/WavefrontProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/wavefront/WavefrontProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,7 @@ * @author Glenn Oppegard * @since 3.0.0 */ -@ConfigurationProperties(prefix = "management.wavefront") +@ConfigurationProperties("management.wavefront") public class WavefrontProperties { /** diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/exchanges/HttpExchangesAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/exchanges/HttpExchangesAutoConfiguration.java index ea79c24e168f..ab33137281a3 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/exchanges/HttpExchangesAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/exchanges/HttpExchangesAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,8 +23,8 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.boot.context.properties.EnableConfigurationProperties; @@ -40,7 +40,7 @@ */ @AutoConfiguration @ConditionalOnWebApplication -@ConditionalOnProperty(prefix = "management.httpexchanges.recording", name = "enabled", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "management.httpexchanges.recording.enabled", matchIfMissing = true) @ConditionalOnBean(HttpExchangeRepository.class) @EnableConfigurationProperties(HttpExchangesProperties.class) public class HttpExchangesAutoConfiguration { diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/exchanges/HttpExchangesProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/exchanges/HttpExchangesProperties.java index b4eb1c1df3fc..cfac338e992b 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/exchanges/HttpExchangesProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/exchanges/HttpExchangesProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "management.httpexchanges") +@ConfigurationProperties("management.httpexchanges") public class HttpExchangesProperties { private final Recording recording = new Recording(); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfiguration.java index d28ec5d41561..36dd324bcf32 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,27 +16,33 @@ package org.springframework.boot.actuate.autoconfigure.web.reactive; +import java.io.File; import java.util.Collections; import java.util.Map; +import org.apache.catalina.Valve; +import org.apache.catalina.valves.AccessLogValve; +import org.eclipse.jetty.server.CustomRequestLog; +import org.eclipse.jetty.server.RequestLog; +import org.eclipse.jetty.server.RequestLogWriter; +import org.eclipse.jetty.server.Server; + import org.springframework.beans.factory.ListableBeanFactory; import org.springframework.boot.actuate.autoconfigure.web.ManagementContextConfiguration; import org.springframework.boot.actuate.autoconfigure.web.ManagementContextType; import org.springframework.boot.actuate.autoconfigure.web.server.ManagementServerProperties; import org.springframework.boot.actuate.autoconfigure.web.server.ManagementWebServerFactoryCustomizer; +import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; -import org.springframework.boot.autoconfigure.web.embedded.JettyVirtualThreadsWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.embedded.JettyWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.embedded.NettyWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.embedded.TomcatVirtualThreadsWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.embedded.TomcatWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.embedded.UndertowWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.reactive.ReactiveWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.reactive.TomcatReactiveWebServerFactoryCustomizer; -import org.springframework.boot.web.reactive.server.ConfigurableReactiveWebServerFactory; +import org.springframework.boot.web.embedded.jetty.JettyReactiveWebServerFactory; +import org.springframework.boot.web.embedded.tomcat.TomcatReactiveWebServerFactory; +import org.springframework.boot.web.embedded.undertow.UndertowReactiveWebServerFactory; +import org.springframework.boot.web.server.ConfigurableWebServerFactory; +import org.springframework.boot.web.server.WebServerFactoryCustomizer; import org.springframework.context.ApplicationContext; import org.springframework.context.annotation.Bean; +import org.springframework.core.Ordered; import org.springframework.http.server.reactive.ContextPathCompositeHandler; import org.springframework.http.server.reactive.HttpHandler; import org.springframework.util.StringUtils; @@ -50,6 +56,7 @@ * * @author Andy Wilkinson * @author Phillip Webb + * @author Moritz Halbritter * @since 2.0.0 */ @EnableWebFlux @@ -58,9 +65,9 @@ public class ReactiveManagementChildContextConfiguration { @Bean - public ReactiveManagementWebServerFactoryCustomizer reactiveManagementWebServerFactoryCustomizer( + public ManagementWebServerFactoryCustomizer reactiveManagementWebServerFactoryCustomizer( ListableBeanFactory beanFactory) { - return new ReactiveManagementWebServerFactoryCustomizer(beanFactory); + return new ManagementWebServerFactoryCustomizer<>(beanFactory); } @Bean @@ -73,15 +80,125 @@ public HttpHandler httpHandler(ApplicationContext applicationContext, Management return httpHandler; } - static class ReactiveManagementWebServerFactoryCustomizer - extends ManagementWebServerFactoryCustomizer { + @Bean + @ConditionalOnClass(name = "io.undertow.Undertow") + UndertowAccessLogCustomizer undertowManagementAccessLogCustomizer(ManagementServerProperties properties) { + return new UndertowAccessLogCustomizer(properties); + } + + @Bean + @ConditionalOnClass(name = "org.apache.catalina.valves.AccessLogValve") + TomcatAccessLogCustomizer tomcatManagementAccessLogCustomizer(ManagementServerProperties properties) { + return new TomcatAccessLogCustomizer(properties); + } + + @Bean + @ConditionalOnClass(name = "org.eclipse.jetty.server.Server") + JettyAccessLogCustomizer jettyManagementAccessLogCustomizer(ManagementServerProperties properties) { + return new JettyAccessLogCustomizer(properties); + } + + abstract static class AccessLogCustomizer implements Ordered { + + private final String prefix; + + AccessLogCustomizer(String prefix) { + this.prefix = prefix; + } + + protected String customizePrefix(String existingPrefix) { + if (this.prefix == null) { + return existingPrefix; + } + if (existingPrefix == null) { + return this.prefix; + } + if (existingPrefix.startsWith(this.prefix)) { + return existingPrefix; + } + return this.prefix + existingPrefix; + } + + @Override + public int getOrder() { + return 1; + } + + } + + static class TomcatAccessLogCustomizer extends AccessLogCustomizer + implements WebServerFactoryCustomizer { + + TomcatAccessLogCustomizer(ManagementServerProperties properties) { + super(properties.getTomcat().getAccesslog().getPrefix()); + } + + @Override + public void customize(TomcatReactiveWebServerFactory factory) { + AccessLogValve accessLogValve = findAccessLogValve(factory); + if (accessLogValve == null) { + return; + } + accessLogValve.setPrefix(customizePrefix(accessLogValve.getPrefix())); + } + + private AccessLogValve findAccessLogValve(TomcatReactiveWebServerFactory factory) { + for (Valve engineValve : factory.getEngineValves()) { + if (engineValve instanceof AccessLogValve accessLogValve) { + return accessLogValve; + } + } + return null; + } + + } + + static class UndertowAccessLogCustomizer extends AccessLogCustomizer + implements WebServerFactoryCustomizer { + + UndertowAccessLogCustomizer(ManagementServerProperties properties) { + super(properties.getUndertow().getAccesslog().getPrefix()); + } + + @Override + public void customize(UndertowReactiveWebServerFactory factory) { + factory.setAccessLogPrefix(customizePrefix(factory.getAccessLogPrefix())); + } + + } + + static class JettyAccessLogCustomizer extends AccessLogCustomizer + implements WebServerFactoryCustomizer { + + JettyAccessLogCustomizer(ManagementServerProperties properties) { + super(properties.getJetty().getAccesslog().getPrefix()); + } + + @Override + public void customize(JettyReactiveWebServerFactory factory) { + factory.addServerCustomizers(this::customizeServer); + } + + private void customizeServer(Server server) { + RequestLog requestLog = server.getRequestLog(); + if (requestLog instanceof CustomRequestLog customRequestLog) { + customizeRequestLog(customRequestLog); + } + } + + private void customizeRequestLog(CustomRequestLog requestLog) { + if (requestLog.getWriter() instanceof RequestLogWriter requestLogWriter) { + customizeRequestLogWriter(requestLogWriter); + } + } - ReactiveManagementWebServerFactoryCustomizer(ListableBeanFactory beanFactory) { - super(beanFactory, ReactiveWebServerFactoryCustomizer.class, TomcatWebServerFactoryCustomizer.class, - TomcatReactiveWebServerFactoryCustomizer.class, - TomcatVirtualThreadsWebServerFactoryCustomizer.class, JettyWebServerFactoryCustomizer.class, - JettyVirtualThreadsWebServerFactoryCustomizer.class, UndertowWebServerFactoryCustomizer.class, - NettyWebServerFactoryCustomizer.class); + private void customizeRequestLogWriter(RequestLogWriter writer) { + String filename = writer.getFileName(); + if (StringUtils.hasLength(filename)) { + File file = new File(filename); + file = new File(file.getParentFile(), customizePrefix(file.getName())); + writer.setFilename(file.getPath()); + } } } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementContextAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementContextAutoConfiguration.java index 25004ecebce1..5afb49200ba2 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementContextAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementContextAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,6 +25,7 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; +import org.springframework.boot.autoconfigure.web.embedded.EmbeddedWebServerFactoryCustomizerAutoConfiguration; import org.springframework.boot.autoconfigure.web.reactive.ReactiveWebServerFactoryAutoConfiguration; import org.springframework.boot.web.reactive.server.ReactiveWebServerFactory; import org.springframework.context.annotation.Bean; @@ -44,7 +45,8 @@ public class ReactiveManagementContextAutoConfiguration { @Bean public static ManagementContextFactory reactiveWebChildContextFactory() { return new ManagementContextFactory(WebApplicationType.REACTIVE, ReactiveWebServerFactory.class, - ReactiveWebServerFactoryAutoConfiguration.class); + ReactiveWebServerFactoryAutoConfiguration.class, + EmbeddedWebServerFactoryCustomizerAutoConfiguration.class); } } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementServerProperties.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementServerProperties.java index 297a885bd6b6..527b5f1bbeb8 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementServerProperties.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementServerProperties.java @@ -30,6 +30,7 @@ * @author Dave Syer * @author Stephane Nicoll * @author Vedran Pavic + * @author Moritz Halbritter * @since 2.0.0 * @see ServerProperties */ @@ -57,6 +58,12 @@ public class ManagementServerProperties { @NestedConfigurationProperty private Ssl ssl; + private final Jetty jetty = new Jetty(); + + private final Tomcat tomcat = new Tomcat(); + + private final Undertow undertow = new Undertow(); + /** * Returns the management port or {@code null} if the * {@link ServerProperties#getPort() server port} should be used. @@ -101,6 +108,18 @@ public void setSsl(Ssl ssl) { this.ssl = ssl; } + public Jetty getJetty() { + return this.jetty; + } + + public Tomcat getTomcat() { + return this.tomcat; + } + + public Undertow getUndertow() { + return this.undertow; + } + private String cleanBasePath(String basePath) { String candidate = null; if (StringUtils.hasLength(basePath)) { @@ -117,4 +136,51 @@ private String cleanBasePath(String basePath) { return candidate; } + public static class Jetty { + + private final Accesslog accesslog = new Accesslog(); + + public Accesslog getAccesslog() { + return this.accesslog; + } + + } + + public static class Tomcat { + + private final Accesslog accesslog = new Accesslog(); + + public Accesslog getAccesslog() { + return this.accesslog; + } + + } + + public static class Undertow { + + private final Accesslog accesslog = new Accesslog(); + + public Accesslog getAccesslog() { + return this.accesslog; + } + + } + + public static class Accesslog { + + /** + * Management log file name prefix. + */ + private String prefix = "management_"; + + public String getPrefix() { + return this.prefix; + } + + public void setPrefix(String prefix) { + this.prefix = prefix; + } + + } + } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementWebServerFactoryCustomizer.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementWebServerFactoryCustomizer.java index aa13066f1419..265eaf6d5ffb 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementWebServerFactoryCustomizer.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementWebServerFactoryCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,7 @@ * @author Andy Wilkinson * @since 2.0.0 */ -public abstract class ManagementWebServerFactoryCustomizer +public class ManagementWebServerFactoryCustomizer implements WebServerFactoryCustomizer, Ordered { private final ListableBeanFactory beanFactory; @@ -48,12 +48,24 @@ public abstract class ManagementWebServerFactoryCustomizer>... customizerClasses) { this.beanFactory = beanFactory; this.customizerClasses = customizerClasses; } + /** + * Creates a new customizer that will retrieve beans using the given + * {@code beanFactory}. + * @param beanFactory the bean factory to use + * @since 3.5.0 + */ + public ManagementWebServerFactoryCustomizer(ListableBeanFactory beanFactory) { + this.beanFactory = beanFactory; + this.customizerClasses = null; + } + @Override public int getOrder() { return 0; @@ -65,7 +77,9 @@ public final void customize(T factory) { .beanOfTypeIncludingAncestors(this.beanFactory, ManagementServerProperties.class); // Customize as per the parent context first (so e.g. the access logs go to // the same place) - customizeSameAsParentContext(factory); + if (this.customizerClasses != null) { + customizeSameAsParentContext(factory); + } // Then reset the error pages factory.setErrorPages(Collections.emptySet()); // and add the management-specific bits diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ManagementErrorEndpoint.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ManagementErrorEndpoint.java index 2759ca4d1345..f4a814d47ecf 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ManagementErrorEndpoint.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ManagementErrorEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,8 +47,8 @@ public class ManagementErrorEndpoint { private final ErrorProperties errorProperties; public ManagementErrorEndpoint(ErrorAttributes errorAttributes, ErrorProperties errorProperties) { - Assert.notNull(errorAttributes, "ErrorAttributes must not be null"); - Assert.notNull(errorProperties, "ErrorProperties must not be null"); + Assert.notNull(errorAttributes, "'errorAttributes' must not be null"); + Assert.notNull(errorProperties, "'errorProperties' must not be null"); this.errorAttributes = errorAttributes; this.errorProperties = errorProperties; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementChildContextConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementChildContextConfiguration.java index 71beda39b6ba..fefe79cb1a02 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementChildContextConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementChildContextConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,14 +39,7 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.boot.autoconfigure.condition.SearchStrategy; import org.springframework.boot.autoconfigure.web.ServerProperties; -import org.springframework.boot.autoconfigure.web.embedded.JettyVirtualThreadsWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.embedded.JettyWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.embedded.TomcatVirtualThreadsWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.embedded.TomcatWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.embedded.UndertowWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.servlet.ServletWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.servlet.TomcatServletWebServerFactoryCustomizer; -import org.springframework.boot.autoconfigure.web.servlet.UndertowServletWebServerFactoryCustomizer; +import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.boot.web.embedded.jetty.JettyServletWebServerFactory; import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory; import org.springframework.boot.web.embedded.undertow.UndertowServletWebServerFactory; @@ -70,9 +63,11 @@ * @author Andy Wilkinson * @author Eddú Meléndez * @author Phillip Webb + * @author Moritz Halbritter */ @ManagementContextConfiguration(value = ManagementContextType.CHILD, proxyBeanMethods = false) @ConditionalOnWebApplication(type = Type.SERVLET) +@EnableConfigurationProperties(ManagementServerProperties.class) class ServletManagementChildContextConfiguration { @Bean @@ -83,20 +78,20 @@ ServletManagementWebServerFactoryCustomizer servletManagementWebServerFactoryCus @Bean @ConditionalOnClass(name = "io.undertow.Undertow") - UndertowAccessLogCustomizer undertowManagementAccessLogCustomizer() { - return new UndertowAccessLogCustomizer(); + UndertowAccessLogCustomizer undertowManagementAccessLogCustomizer(ManagementServerProperties properties) { + return new UndertowAccessLogCustomizer(properties); } @Bean @ConditionalOnClass(name = "org.apache.catalina.valves.AccessLogValve") - TomcatAccessLogCustomizer tomcatManagementAccessLogCustomizer() { - return new TomcatAccessLogCustomizer(); + TomcatAccessLogCustomizer tomcatManagementAccessLogCustomizer(ManagementServerProperties properties) { + return new TomcatAccessLogCustomizer(properties); } @Bean @ConditionalOnClass(name = "org.eclipse.jetty.server.Server") - JettyAccessLogCustomizer jettyManagementAccessLogCustomizer() { - return new JettyAccessLogCustomizer(); + JettyAccessLogCustomizer jettyManagementAccessLogCustomizer(ManagementServerProperties properties) { + return new JettyAccessLogCustomizer(properties); } @Configuration(proxyBeanMethods = false) @@ -123,10 +118,7 @@ static class ServletManagementWebServerFactoryCustomizer extends ManagementWebServerFactoryCustomizer { ServletManagementWebServerFactoryCustomizer(ListableBeanFactory beanFactory) { - super(beanFactory, ServletWebServerFactoryCustomizer.class, TomcatServletWebServerFactoryCustomizer.class, - TomcatWebServerFactoryCustomizer.class, TomcatVirtualThreadsWebServerFactoryCustomizer.class, - JettyWebServerFactoryCustomizer.class, JettyVirtualThreadsWebServerFactoryCustomizer.class, - UndertowServletWebServerFactoryCustomizer.class, UndertowWebServerFactoryCustomizer.class); + super(beanFactory); } @Override @@ -145,14 +137,23 @@ private String getContextPath(ManagementServerProperties managementServerPropert abstract static class AccessLogCustomizer implements Ordered { - private static final String MANAGEMENT_PREFIX = "management_"; + private final String prefix; - protected String customizePrefix(String prefix) { - prefix = (prefix != null) ? prefix : ""; - if (prefix.startsWith(MANAGEMENT_PREFIX)) { - return prefix; + AccessLogCustomizer(String prefix) { + this.prefix = prefix; + } + + protected String customizePrefix(String existingPrefix) { + if (this.prefix == null) { + return existingPrefix; + } + if (existingPrefix == null) { + return this.prefix; } - return MANAGEMENT_PREFIX + prefix; + if (existingPrefix.startsWith(this.prefix)) { + return existingPrefix; + } + return this.prefix + existingPrefix; } @Override @@ -165,6 +166,10 @@ public int getOrder() { static class TomcatAccessLogCustomizer extends AccessLogCustomizer implements WebServerFactoryCustomizer { + TomcatAccessLogCustomizer(ManagementServerProperties properties) { + super(properties.getTomcat().getAccesslog().getPrefix()); + } + @Override public void customize(TomcatServletWebServerFactory factory) { AccessLogValve accessLogValve = findAccessLogValve(factory); @@ -188,6 +193,10 @@ private AccessLogValve findAccessLogValve(TomcatServletWebServerFactory factory) static class UndertowAccessLogCustomizer extends AccessLogCustomizer implements WebServerFactoryCustomizer { + UndertowAccessLogCustomizer(ManagementServerProperties properties) { + super(properties.getUndertow().getAccesslog().getPrefix()); + } + @Override public void customize(UndertowServletWebServerFactory factory) { factory.setAccessLogPrefix(customizePrefix(factory.getAccessLogPrefix())); @@ -198,6 +207,10 @@ public void customize(UndertowServletWebServerFactory factory) { static class JettyAccessLogCustomizer extends AccessLogCustomizer implements WebServerFactoryCustomizer { + JettyAccessLogCustomizer(ManagementServerProperties properties) { + super(properties.getJetty().getAccesslog().getPrefix()); + } + @Override public void customize(JettyServletWebServerFactory factory) { factory.addServerCustomizers(this::customizeServer); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementContextAutoConfiguration.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementContextAutoConfiguration.java index 7b9fedaa4be3..854ecc26b7bc 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementContextAutoConfiguration.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementContextAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,10 +23,11 @@ import org.springframework.boot.actuate.autoconfigure.web.ManagementContextFactory; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; +import org.springframework.boot.autoconfigure.web.embedded.EmbeddedWebServerFactoryCustomizerAutoConfiguration; import org.springframework.boot.autoconfigure.web.servlet.ServletWebServerFactoryAutoConfiguration; import org.springframework.boot.web.servlet.filter.ApplicationContextHeaderFilter; import org.springframework.boot.web.servlet.server.ServletWebServerFactory; @@ -49,7 +50,8 @@ public class ServletManagementContextAutoConfiguration { @Bean public static ManagementContextFactory servletWebChildContextFactory() { return new ManagementContextFactory(WebApplicationType.SERVLET, ServletWebServerFactory.class, - ServletWebServerFactoryAutoConfiguration.class); + ServletWebServerFactoryAutoConfiguration.class, + EmbeddedWebServerFactoryCustomizerAutoConfiguration.class); } @Bean @@ -60,7 +62,7 @@ public ManagementServletContext managementServletContext(WebEndpointProperties p // Put Servlets and Filters in their own nested class so they don't force early // instantiation of ManagementServerProperties. @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "management.server", name = "add-application-context-header", havingValue = "true") + @ConditionalOnBooleanProperty("management.server.add-application-context-header") protected static class ApplicationContextFilterConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/resources/META-INF/additional-spring-configuration-metadata.json b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/resources/META-INF/additional-spring-configuration-metadata.json index 9ef8ab6526a0..a6e79482380e 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/resources/META-INF/additional-spring-configuration-metadata.json +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/resources/META-INF/additional-spring-configuration-metadata.json @@ -64,7 +64,7 @@ }, { "name": "management.endpoints.access.max-permitted", - "description": "The maximum level of endpoint access that is permitted. Caps an endpoint's individual access level (management.endpoint..access) and the default access (management.endpoints.access.default).'", + "description": "Maximum level of endpoint access that is permitted. Caps an endpoint's individual access level (management.endpoint..access) and the default access (management.endpoints.access.default).'", "defaultValue": "unrestricted" }, { @@ -1926,7 +1926,7 @@ }, { "name": "management.metrics.graphql.autotime.enabled", - "description": "Whether to automatically time GraphQL requests.", + "description": "Whether to automatically time web client requests.", "defaultValue": true, "deprecation": { "level": "error", @@ -2083,6 +2083,14 @@ "type": "java.lang.Boolean", "description": "Whether auto-configuration of tracing is enabled to export OTLP traces." }, + { + "name": "management.promethus.metrics.export.pushgateway.base-url", + "type": "java.lang.String", + "deprecation": { + "level": "error", + "replacement": "management.prometheus.metrics.export.pushgateway.address" + } + }, { "name": "management.server.add-application-context-header", "type": "java.lang.Boolean", @@ -2099,7 +2107,7 @@ }, { "name": "management.server.ssl.bundle", - "description": "The name of a configured SSL bundle." + "description": "Name of a configured SSL bundle." }, { "name": "management.server.ssl.certificate", @@ -2239,14 +2247,6 @@ "name": "management.zipkin.tracing.export.enabled", "type": "java.lang.Boolean", "description": "Whether auto-configuration of tracing is enabled to export Zipkin traces." - }, - { - "name": "micrometer.observations.annotations.enabled", - "type": "java.lang.Boolean", - "deprecation": { - "level": "error", - "replacement": "management.observations.annotations.enabled" - } } ], "hints": [ diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports index 56f12f9b44a5..201b9cfdaeda 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports @@ -65,7 +65,6 @@ org.springframework.boot.actuate.autoconfigure.metrics.export.kairos.KairosMetri org.springframework.boot.actuate.autoconfigure.metrics.export.newrelic.NewRelicMetricsExportAutoConfiguration org.springframework.boot.actuate.autoconfigure.metrics.export.otlp.OtlpMetricsExportAutoConfiguration org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus.PrometheusMetricsExportAutoConfiguration -org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus.PrometheusSimpleclientMetricsExportAutoConfiguration org.springframework.boot.actuate.autoconfigure.metrics.export.signalfx.SignalFxMetricsExportAutoConfiguration org.springframework.boot.actuate.autoconfigure.metrics.export.simple.SimpleMetricsExportAutoConfiguration org.springframework.boot.actuate.autoconfigure.metrics.export.stackdriver.StackdriverMetricsExportAutoConfiguration @@ -105,6 +104,7 @@ org.springframework.boot.actuate.autoconfigure.security.servlet.ManagementWebSec org.springframework.boot.actuate.autoconfigure.session.SessionsEndpointAutoConfiguration org.springframework.boot.actuate.autoconfigure.startup.StartupEndpointAutoConfiguration org.springframework.boot.actuate.autoconfigure.ssl.SslHealthContributorAutoConfiguration +org.springframework.boot.actuate.autoconfigure.ssl.SslObservabilityAutoConfiguration org.springframework.boot.actuate.autoconfigure.system.DiskSpaceHealthContributorAutoConfiguration org.springframework.boot.actuate.autoconfigure.tracing.BraveAutoConfiguration org.springframework.boot.actuate.autoconfigure.tracing.MicrometerTracingAutoConfiguration @@ -112,7 +112,6 @@ org.springframework.boot.actuate.autoconfigure.tracing.NoopTracerAutoConfigurati org.springframework.boot.actuate.autoconfigure.tracing.OpenTelemetryTracingAutoConfiguration org.springframework.boot.actuate.autoconfigure.tracing.otlp.OtlpTracingAutoConfiguration org.springframework.boot.actuate.autoconfigure.tracing.prometheus.PrometheusExemplarsAutoConfiguration -org.springframework.boot.actuate.autoconfigure.tracing.prometheus.PrometheusSimpleclientExemplarsAutoConfiguration org.springframework.boot.actuate.autoconfigure.tracing.wavefront.WavefrontTracingAutoConfiguration org.springframework.boot.actuate.autoconfigure.tracing.zipkin.ZipkinAutoConfiguration org.springframework.boot.actuate.autoconfigure.wavefront.WavefrontAutoConfiguration diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/availability/AvailabilityProbesHealthEndpointGroupsTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/availability/AvailabilityProbesHealthEndpointGroupsTests.java index ae5dfc538455..caae873f31a1 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/availability/AvailabilityProbesHealthEndpointGroupsTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/availability/AvailabilityProbesHealthEndpointGroupsTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -52,7 +52,7 @@ void setup() { @Test void createWhenGroupsIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new AvailabilityProbesHealthEndpointGroups(null, false)) - .withMessage("Groups must not be null"); + .withMessage("'groups' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/CloudFoundryWebFluxEndpointHandlerMappingTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/CloudFoundryWebFluxEndpointHandlerMappingTests.java index f6a8d1cff882..0ab3692ac09c 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/CloudFoundryWebFluxEndpointHandlerMappingTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/CloudFoundryWebFluxEndpointHandlerMappingTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ void shouldRegisterHints() { RuntimeHints runtimeHints = new RuntimeHints(); new CloudFoundryWebFluxEndpointHandlerMappingRuntimeHints().registerHints(runtimeHints, getClass().getClassLoader()); - assertThat(RuntimeHintsPredicates.reflection().onMethod(CloudFoundryLinksHandler.class, "links")) + assertThat(RuntimeHintsPredicates.reflection().onMethod(CloudFoundryLinksHandler.class, "links").invoke()) .accepts(runtimeHints); assertThat(RuntimeHintsPredicates.reflection().onType(Link.class)).accepts(runtimeHints); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundryActuatorAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundryActuatorAutoConfigurationTests.java index 2a0e7adceea7..e020ace3c1ac 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundryActuatorAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/reactive/ReactiveCloudFoundryActuatorAutoConfigurationTests.java @@ -27,6 +27,7 @@ import okhttp3.mockwebserver.MockResponse; import okhttp3.mockwebserver.MockWebServer; +import org.assertj.core.api.InstanceOfAssertFactories; import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.Test; import reactor.netty.http.HttpResources; @@ -43,7 +44,6 @@ import org.springframework.boot.actuate.endpoint.EndpointId; import org.springframework.boot.actuate.endpoint.annotation.Endpoint; import org.springframework.boot.actuate.endpoint.annotation.ReadOperation; -import org.springframework.boot.actuate.endpoint.web.EndpointMapping; import org.springframework.boot.actuate.endpoint.web.ExposableWebEndpoint; import org.springframework.boot.actuate.endpoint.web.WebOperation; import org.springframework.boot.actuate.endpoint.web.WebOperationRequestPredicate; @@ -74,7 +74,6 @@ import org.springframework.security.core.userdetails.User; import org.springframework.security.web.server.SecurityWebFilterChain; import org.springframework.security.web.server.WebFilterChainProxy; -import org.springframework.test.util.ReflectionTestUtils; import org.springframework.test.web.reactive.server.WebTestClient; import org.springframework.web.cors.CorsConfiguration; import org.springframework.web.reactive.function.client.WebClient; @@ -121,16 +120,16 @@ void cloudFoundryPlatformActive() { "vcap.application.cf_api:https://my-cloud-controller.com") .run((context) -> { CloudFoundryWebFluxEndpointHandlerMapping handlerMapping = getHandlerMapping(context); - EndpointMapping endpointMapping = (EndpointMapping) ReflectionTestUtils.getField(handlerMapping, - "endpointMapping"); - assertThat(endpointMapping.getPath()).isEqualTo("/cloudfoundryapplication"); - CorsConfiguration corsConfiguration = (CorsConfiguration) ReflectionTestUtils.getField(handlerMapping, - "corsConfiguration"); - assertThat(corsConfiguration.getAllowedOrigins()).contains("*"); - assertThat(corsConfiguration.getAllowedMethods()) - .containsAll(Arrays.asList(HttpMethod.GET.name(), HttpMethod.POST.name())); - assertThat(corsConfiguration.getAllowedHeaders()) - .containsAll(Arrays.asList("Authorization", "X-Cf-App-Instance", "Content-Type")); + assertThat(handlerMapping).extracting("endpointMapping.path").isEqualTo("/cloudfoundryapplication"); + assertThat(handlerMapping) + .extracting("corsConfiguration", InstanceOfAssertFactories.type(CorsConfiguration.class)) + .satisfies((corsConfiguration) -> { + assertThat(corsConfiguration.getAllowedOrigins()).contains("*"); + assertThat(corsConfiguration.getAllowedMethods()) + .containsAll(Arrays.asList(HttpMethod.GET.name(), HttpMethod.POST.name())); + assertThat(corsConfiguration.getAllowedHeaders()) + .containsAll(Arrays.asList("Authorization", "X-Cf-App-Instance", "Content-Type")); + }); }); } @@ -150,12 +149,8 @@ void cloudFoundryPlatformActiveSetsApplicationId() { this.contextRunner .withPropertyValues("VCAP_APPLICATION:---", "vcap.application.application_id:my-app-id", "vcap.application.cf_api:https://my-cloud-controller.com") - .run((context) -> { - CloudFoundryWebFluxEndpointHandlerMapping handlerMapping = getHandlerMapping(context); - Object interceptor = ReflectionTestUtils.getField(handlerMapping, "securityInterceptor"); - String applicationId = (String) ReflectionTestUtils.getField(interceptor, "applicationId"); - assertThat(applicationId).isEqualTo("my-app-id"); - }); + .run((context) -> assertThat(getHandlerMapping(context)).extracting("securityInterceptor.applicationId") + .isEqualTo("my-app-id")); } @Test @@ -163,28 +158,18 @@ void cloudFoundryPlatformActiveSetsCloudControllerUrl() { this.contextRunner .withPropertyValues("VCAP_APPLICATION:---", "vcap.application.application_id:my-app-id", "vcap.application.cf_api:https://my-cloud-controller.com") - .run((context) -> { - CloudFoundryWebFluxEndpointHandlerMapping handlerMapping = getHandlerMapping(context); - Object interceptor = ReflectionTestUtils.getField(handlerMapping, "securityInterceptor"); - Object interceptorSecurityService = ReflectionTestUtils.getField(interceptor, - "cloudFoundrySecurityService"); - String cloudControllerUrl = (String) ReflectionTestUtils.getField(interceptorSecurityService, - "cloudControllerUrl"); - assertThat(cloudControllerUrl).isEqualTo("https://my-cloud-controller.com"); - }); + .run((context) -> assertThat(getHandlerMapping(context)) + .extracting("securityInterceptor.cloudFoundrySecurityService.cloudControllerUrl") + .isEqualTo("https://my-cloud-controller.com")); } @Test void cloudFoundryPlatformActiveAndCloudControllerUrlNotPresent() { this.contextRunner.withPropertyValues("VCAP_APPLICATION:---", "vcap.application.application_id:my-app-id") - .run((context) -> { - CloudFoundryWebFluxEndpointHandlerMapping handlerMapping = context.getBean( - "cloudFoundryWebFluxEndpointHandlerMapping", CloudFoundryWebFluxEndpointHandlerMapping.class); - Object securityInterceptor = ReflectionTestUtils.getField(handlerMapping, "securityInterceptor"); - Object interceptorSecurityService = ReflectionTestUtils.getField(securityInterceptor, - "cloudFoundrySecurityService"); - assertThat(interceptorSecurityService).isNull(); - }); + .run((context) -> assertThat(context.getBean("cloudFoundryWebFluxEndpointHandlerMapping", + CloudFoundryWebFluxEndpointHandlerMapping.class)) + .extracting("securityInterceptor.cloudFoundrySecurityService") + .isNull()); } @Test @@ -194,30 +179,30 @@ void cloudFoundryPathsIgnoredBySpringSecurity() { .withPropertyValues("VCAP_APPLICATION:---", "vcap.application.application_id:my-app-id", "vcap.application.cf_api:https://my-cloud-controller.com") .run((context) -> { - WebFilterChainProxy chainProxy = context.getBean(WebFilterChainProxy.class); - List filters = (List) ReflectionTestUtils - .getField(chainProxy, "filters"); - Boolean cfBaseRequestMatches = getMatches(filters, BASE_PATH); - Boolean cfBaseWithTrailingSlashRequestMatches = getMatches(filters, BASE_PATH + "/"); - Boolean cfRequestMatches = getMatches(filters, BASE_PATH + "/test"); - Boolean cfRequestWithAdditionalPathMatches = getMatches(filters, BASE_PATH + "/test/a"); - Boolean otherCfRequestMatches = getMatches(filters, BASE_PATH + "/other-path"); - Boolean otherRequestMatches = getMatches(filters, "/some-other-path"); - assertThat(cfBaseRequestMatches).isTrue(); - assertThat(cfBaseWithTrailingSlashRequestMatches).isTrue(); - assertThat(cfRequestMatches).isTrue(); - assertThat(cfRequestWithAdditionalPathMatches).isTrue(); - assertThat(otherCfRequestMatches).isFalse(); - assertThat(otherRequestMatches).isFalse(); - otherRequestMatches = filters.get(1) - .matches(MockServerWebExchange.from(MockServerHttpRequest.get("/some-other-path").build())) - .block(Duration.ofSeconds(30)); - assertThat(otherRequestMatches).isTrue(); + assertThat(context.getBean(WebFilterChainProxy.class)) + .extracting("filters", InstanceOfAssertFactories.list(SecurityWebFilterChain.class)) + .satisfies((filters) -> { + Boolean cfBaseRequestMatches = getMatches(filters, BASE_PATH); + Boolean cfBaseWithTrailingSlashRequestMatches = getMatches(filters, BASE_PATH + "/"); + Boolean cfRequestMatches = getMatches(filters, BASE_PATH + "/test"); + Boolean cfRequestWithAdditionalPathMatches = getMatches(filters, BASE_PATH + "/test/a"); + Boolean otherCfRequestMatches = getMatches(filters, BASE_PATH + "/other-path"); + Boolean otherRequestMatches = getMatches(filters, "/some-other-path"); + assertThat(cfBaseRequestMatches).isTrue(); + assertThat(cfBaseWithTrailingSlashRequestMatches).isTrue(); + assertThat(cfRequestMatches).isTrue(); + assertThat(cfRequestWithAdditionalPathMatches).isTrue(); + assertThat(otherCfRequestMatches).isFalse(); + assertThat(otherRequestMatches).isFalse(); + otherRequestMatches = filters.get(1) + .matches(MockServerWebExchange.from(MockServerHttpRequest.get("/some-other-path").build())) + .block(Duration.ofSeconds(30)); + assertThat(otherRequestMatches).isTrue(); + }); }); - } - private static Boolean getMatches(List filters, String urlTemplate) { + private static Boolean getMatches(List filters, String urlTemplate) { return filters.get(0) .matches(MockServerWebExchange.from(MockServerHttpRequest.get(urlTemplate).build())) .block(Duration.ofSeconds(30)); @@ -322,20 +307,17 @@ void skipSslValidation() throws IOException { .withPropertyValues("VCAP_APPLICATION:---", "vcap.application.application_id:my-app-id", "vcap.application.cf_api:https://my-cloud-controller.com", "management.cloudfoundry.skip-ssl-validation:true") - .run((context) -> { - CloudFoundryWebFluxEndpointHandlerMapping handlerMapping = getHandlerMapping(context); - Object interceptor = ReflectionTestUtils.getField(handlerMapping, "securityInterceptor"); - Object interceptorSecurityService = ReflectionTestUtils.getField(interceptor, - "cloudFoundrySecurityService"); - WebClient webClient = (WebClient) ReflectionTestUtils.getField(interceptorSecurityService, - "webClient"); - ResponseEntity response = webClient.get() - .uri(server.url("/").uri()) - .retrieve() - .toBodilessEntity() - .block(Duration.ofSeconds(30)); - assertThat(response.getStatusCode()).isEqualTo(HttpStatusCode.valueOf(204)); - }); + .run((context) -> assertThat(getHandlerMapping(context)) + .extracting("securityInterceptor.cloudFoundrySecurityService.webClient", + InstanceOfAssertFactories.type(WebClient.class)) + .satisfies((webClient) -> { + ResponseEntity response = webClient.get() + .uri(server.url("/").uri()) + .retrieve() + .toBodilessEntity() + .block(Duration.ofSeconds(30)); + assertThat(response.getStatusCode()).isEqualTo(HttpStatusCode.valueOf(204)); + })); } } @@ -351,21 +333,16 @@ void sslValidationNotSkippedByDefault() throws IOException { this.contextRunner.withConfiguration(AutoConfigurations.of(HealthEndpointAutoConfiguration.class)) .withPropertyValues("VCAP_APPLICATION:---", "vcap.application.application_id:my-app-id", "vcap.application.cf_api:https://my-cloud-controller.com") - .run((context) -> { - CloudFoundryWebFluxEndpointHandlerMapping handlerMapping = getHandlerMapping(context); - Object interceptor = ReflectionTestUtils.getField(handlerMapping, "securityInterceptor"); - Object interceptorSecurityService = ReflectionTestUtils.getField(interceptor, - "cloudFoundrySecurityService"); - WebClient webClient = (WebClient) ReflectionTestUtils.getField(interceptorSecurityService, - "webClient"); - assertThatExceptionOfType(RuntimeException.class) + .run((context) -> assertThat(getHandlerMapping(context)) + .extracting("securityInterceptor.cloudFoundrySecurityService.webClient", + InstanceOfAssertFactories.type(WebClient.class)) + .satisfies((webClient) -> assertThatExceptionOfType(RuntimeException.class) .isThrownBy(() -> webClient.get() .uri(server.url("/").uri()) .retrieve() .toBodilessEntity() .block(Duration.ofSeconds(30))) - .withCauseInstanceOf(SSLException.class); - }); + .withCauseInstanceOf(SSLException.class))); } } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundryWebEndpointServletHandlerMappingTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundryWebEndpointServletHandlerMappingTests.java index a7446f11e4a5..5ac2cda8476b 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundryWebEndpointServletHandlerMappingTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/cloudfoundry/servlet/CloudFoundryWebEndpointServletHandlerMappingTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ void shouldRegisterHints() { RuntimeHints runtimeHints = new RuntimeHints(); new CloudFoundryWebEndpointServletHandlerMappingRuntimeHints().registerHints(runtimeHints, getClass().getClassLoader()); - assertThat(RuntimeHintsPredicates.reflection().onMethod(CloudFoundryLinksHandler.class, "links")) + assertThat(RuntimeHintsPredicates.reflection().onMethod(CloudFoundryLinksHandler.class, "links").invoke()) .accepts(runtimeHints); assertThat(RuntimeHintsPredicates.reflection().onType(Link.class)).accepts(runtimeHints); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/IncludeExcludeEndpointFilterTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/IncludeExcludeEndpointFilterTests.java index 75737180e4a3..a8e1a0df42cc 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/IncludeExcludeEndpointFilterTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/endpoint/expose/IncludeExcludeEndpointFilterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,28 +45,28 @@ class IncludeExcludeEndpointFilterTests { void createWhenEndpointTypeIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new IncludeExcludeEndpointFilter<>(null, new MockEnvironment(), "foo")) - .withMessageContaining("EndpointType must not be null"); + .withMessageContaining("'endpointType' must not be null"); } @Test void createWhenEnvironmentIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new IncludeExcludeEndpointFilter<>(ExposableEndpoint.class, null, "foo")) - .withMessageContaining("Environment must not be null"); + .withMessageContaining("'environment' must not be null"); } @Test void createWhenPrefixIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new IncludeExcludeEndpointFilter<>(ExposableEndpoint.class, new MockEnvironment(), null)) - .withMessageContaining("Prefix must not be empty"); + .withMessageContaining("'prefix' must not be empty"); } @Test void createWhenPrefixIsEmptyShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new IncludeExcludeEndpointFilter<>(ExposableEndpoint.class, new MockEnvironment(), "")) - .withMessageContaining("Prefix must not be empty"); + .withMessageContaining("'prefix' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/WebEndpointPropertiesTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/WebEndpointPropertiesTests.java index 441d15bf6645..318057ac2d20 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/WebEndpointPropertiesTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/endpoint/web/WebEndpointPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ void basePathShouldBeCleaned() { void basePathMustStartWithSlash() { WebEndpointProperties properties = new WebEndpointProperties(); assertThatIllegalArgumentException().isThrownBy(() -> properties.setBasePath("admin")) - .withMessageContaining("Base path must start with '/' or be empty"); + .withMessageContaining("'basePath' must start with '/' or be empty"); } @Test diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/health/AbstractCompositeHealthContributorConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/health/AbstractCompositeHealthContributorConfigurationTests.java index 80a31f339d25..01281465fd47 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/health/AbstractCompositeHealthContributorConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/health/AbstractCompositeHealthContributorConfigurationTests.java @@ -54,7 +54,7 @@ abstract class AbstractCompositeHealthContributorConfigurationTests beans = Collections.emptyMap(); assertThatIllegalArgumentException().isThrownBy(() -> newComposite().createContributor(beans)) - .withMessage("Beans must not be empty"); + .withMessage("'beans' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/health/HealthEndpointAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/health/HealthEndpointAutoConfigurationTests.java index d1144387ac2a..3a836c878bbb 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/health/HealthEndpointAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/health/HealthEndpointAutoConfigurationTests.java @@ -53,9 +53,7 @@ import org.springframework.boot.actuate.health.StatusAggregator; import org.springframework.boot.actuate.health.SystemHealth; import org.springframework.boot.autoconfigure.AutoConfigurations; -import org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListener; import org.springframework.boot.autoconfigure.web.servlet.DispatcherServletAutoConfiguration; -import org.springframework.boot.logging.LogLevel; import org.springframework.boot.test.context.FilteredClassLoader; import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner; @@ -365,7 +363,6 @@ void additionalJerseyHealthEndpointsPathsTolerateHealthEndpointThatIsNotWebExpos .withClassLoader(new FilteredClassLoader(DispatcherServlet.class)) .withPropertyValues("management.endpoints.web.exposure.exclude=*", "management.endpoints.cloudfoundry.exposure.include=*", "spring.main.cloud-platform=cloud_foundry") - .withInitializer(ConditionEvaluationReportLoggingListener.forLogLevel(LogLevel.INFO)) .run((context) -> { assertThat(context).hasSingleBean(JerseyAdditionalHealthEndpointPathsConfiguration.class); assertThat(context).hasNotFailed(); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthContributorAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthContributorAutoConfigurationTests.java index a5cfebe3a5aa..1416bf220053 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthContributorAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/jdbc/DataSourceHealthContributorAutoConfigurationTests.java @@ -232,7 +232,7 @@ void prototypeDataSourceIsIgnored() { static class DataSourceConfig { @Bean - @ConfigurationProperties(prefix = "spring.datasource.test") + @ConfigurationProperties("spring.datasource.test") DataSource standardDataSource() { return DataSourceBuilder.create() .type(org.apache.tomcat.jdbc.pool.DataSource.class) @@ -249,7 +249,7 @@ DataSource standardDataSource() { static class NonStandardDataSourceConfig { @Bean(defaultCandidate = false) - @ConfigurationProperties(prefix = "spring.datasource.non-default") + @ConfigurationProperties("spring.datasource.non-default") DataSource nonDefaultDataSource() { return DataSourceBuilder.create() .type(org.apache.tomcat.jdbc.pool.DataSource.class) @@ -260,7 +260,7 @@ DataSource nonDefaultDataSource() { } @Bean(autowireCandidate = false) - @ConfigurationProperties(prefix = "spring.datasource.non-autowire") + @ConfigurationProperties("spring.datasource.non-autowire") DataSource nonAutowireDataSource() { return DataSourceBuilder.create() .type(org.apache.tomcat.jdbc.pool.DataSource.class) diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/logging/otlp/OtlpLoggingAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/logging/otlp/OtlpLoggingAutoConfigurationTests.java index 34a70bca0d26..017f3dea555a 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/logging/otlp/OtlpLoggingAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/logging/otlp/OtlpLoggingAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,8 +16,13 @@ package org.springframework.boot.actuate.autoconfigure.logging.otlp; +import java.util.function.Supplier; + +import io.opentelemetry.api.metrics.MeterProvider; import io.opentelemetry.exporter.otlp.http.logs.OtlpHttpLogRecordExporter; +import io.opentelemetry.exporter.otlp.http.logs.OtlpHttpLogRecordExporterBuilder; import io.opentelemetry.exporter.otlp.logs.OtlpGrpcLogRecordExporter; +import io.opentelemetry.exporter.otlp.logs.OtlpGrpcLogRecordExporterBuilder; import io.opentelemetry.sdk.logs.export.LogRecordExporter; import okhttp3.HttpUrl; import org.junit.jupiter.api.Test; @@ -30,6 +35,7 @@ import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.test.util.ReflectionTestUtils; import static org.assertj.core.api.Assertions.assertThat; @@ -37,6 +43,7 @@ * Tests for {@link OtlpLoggingAutoConfiguration}. * * @author Toshiaki Maki + * @author Moritz Halbritter */ class OtlpLoggingAutoConfigurationTests { @@ -118,7 +125,6 @@ void shouldBackOffWhenCustomOtlpLoggingConnectionDetailsIsDefined() { assertThat(otlpHttpLogRecordExporter).extracting("delegate.httpSender.url") .isEqualTo(HttpUrl.get("https://otel.example.com/v1/logs")); }); - } @Test @@ -155,31 +161,74 @@ void shouldUseGrpcExporterIfTransportIsSetToGrpc() { }); } + @Test + @SuppressWarnings("unchecked") + void httpShouldUseMeterProviderIfSet() { + this.contextRunner.withUserConfiguration(MeterProviderConfiguration.class) + .withPropertyValues("management.otlp.logging.endpoint=http://localhost:4318/v1/logs") + .run((context) -> { + OtlpHttpLogRecordExporter otlpHttpLogRecordExporter = context.getBean(OtlpHttpLogRecordExporter.class); + OtlpHttpLogRecordExporterBuilder builder = otlpHttpLogRecordExporter.toBuilder(); + Supplier meterProviderSupplier = (Supplier) ReflectionTestUtils + .getField(ReflectionTestUtils.getField(builder, "delegate"), "meterProviderSupplier"); + assertThat(meterProviderSupplier).isNotNull(); + assertThat(meterProviderSupplier.get()).isSameAs(MeterProviderConfiguration.meterProvider); + }); + } + + @Test + @SuppressWarnings("unchecked") + void grpcShouldUseMeterProviderIfSet() { + this.contextRunner.withUserConfiguration(MeterProviderConfiguration.class) + .withPropertyValues("management.otlp.logging.endpoint=http://localhost:4318/v1/logs", + "management.otlp.logging.transport=grpc") + .run((context) -> { + OtlpGrpcLogRecordExporter otlpGrpcLogRecordExporter = context.getBean(OtlpGrpcLogRecordExporter.class); + OtlpGrpcLogRecordExporterBuilder builder = otlpGrpcLogRecordExporter.toBuilder(); + Supplier meterProviderSupplier = (Supplier) ReflectionTestUtils + .getField(ReflectionTestUtils.getField(builder, "delegate"), "meterProviderSupplier"); + assertThat(meterProviderSupplier).isNotNull(); + assertThat(meterProviderSupplier.get()).isSameAs(MeterProviderConfiguration.meterProvider); + }); + } + + @Configuration(proxyBeanMethods = false) + private static final class MeterProviderConfiguration { + + static final MeterProvider meterProvider = (instrumentationScopeName) -> null; + + @Bean + MeterProvider meterProvider() { + return meterProvider; + } + + } + @Configuration(proxyBeanMethods = false) - public static class CustomHttpExporterConfiguration { + private static final class CustomHttpExporterConfiguration { @Bean - public OtlpHttpLogRecordExporter customOtlpHttpLogRecordExporter() { + OtlpHttpLogRecordExporter customOtlpHttpLogRecordExporter() { return OtlpHttpLogRecordExporter.builder().build(); } } @Configuration(proxyBeanMethods = false) - public static class CustomGrpcExporterConfiguration { + private static final class CustomGrpcExporterConfiguration { @Bean - public OtlpGrpcLogRecordExporter customOtlpGrpcLogRecordExporter() { + OtlpGrpcLogRecordExporter customOtlpGrpcLogRecordExporter() { return OtlpGrpcLogRecordExporter.builder().build(); } } @Configuration(proxyBeanMethods = false) - public static class CustomOtlpLoggingConnectionDetails { + private static final class CustomOtlpLoggingConnectionDetails { @Bean - public OtlpLoggingConnectionDetails customOtlpLoggingConnectionDetails() { + OtlpLoggingConnectionDetails customOtlpLoggingConnectionDetails() { return (transport) -> "https://otel.example.com/v1/logs"; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/JvmMetricsAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/JvmMetricsAutoConfigurationTests.java index 16d1e473894e..491246ce7398 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/JvmMetricsAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/JvmMetricsAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,6 +16,7 @@ package org.springframework.boot.actuate.autoconfigure.metrics; +import io.micrometer.core.instrument.binder.MeterBinder; import io.micrometer.core.instrument.binder.jvm.ClassLoaderMetrics; import io.micrometer.core.instrument.binder.jvm.JvmCompilationMetrics; import io.micrometer.core.instrument.binder.jvm.JvmGcMetrics; @@ -24,7 +25,14 @@ import io.micrometer.core.instrument.binder.jvm.JvmMemoryMetrics; import io.micrometer.core.instrument.binder.jvm.JvmThreadMetrics; import org.junit.jupiter.api.Test; - +import org.junit.jupiter.api.condition.EnabledForJreRange; +import org.junit.jupiter.api.condition.JRE; + +import org.springframework.aot.hint.MemberCategory; +import org.springframework.aot.hint.RuntimeHints; +import org.springframework.aot.hint.TypeReference; +import org.springframework.aot.hint.predicate.RuntimeHintsPredicates; +import org.springframework.beans.BeanUtils; import org.springframework.boot.actuate.autoconfigure.metrics.test.MetricsRun; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.test.context.assertj.AssertableApplicationContext; @@ -32,6 +40,7 @@ import org.springframework.boot.test.context.runner.ContextConsumer; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.util.ClassUtils; import static org.assertj.core.api.Assertions.assertThat; @@ -95,6 +104,36 @@ void allowsCustomJvmCompilationMetricsToBeUsed() { .run(assertMetricsBeans().andThen((context) -> assertThat(context).hasBean("customJvmCompilationMetrics"))); } + @Test + @EnabledForJreRange(min = JRE.JAVA_21) + void autoConfiguresJvmMetricsWithVirtualThreadsMetrics() { + this.contextRunner.run(assertMetricsBeans() + .andThen((context) -> assertThat(context).hasSingleBean(getVirtualThreadMetricsClass()))); + } + + @Test + @EnabledForJreRange(min = JRE.JAVA_21) + void allowCustomVirtualThreadMetricsToBeUsed() { + Class virtualThreadMetricsClass = getVirtualThreadMetricsClass(); + this.contextRunner + .withBean("customVirtualThreadMetrics", virtualThreadMetricsClass, + () -> BeanUtils.instantiateClass(virtualThreadMetricsClass)) + .run(assertMetricsBeans() + .andThen((context) -> assertThat(context).hasSingleBean(getVirtualThreadMetricsClass()) + .hasBean("customVirtualThreadMetrics"))); + } + + @Test + @EnabledForJreRange(min = JRE.JAVA_21) + void shouldRegisterVirtualThreadMetricsRuntimeHints() { + RuntimeHints hints = new RuntimeHints(); + new JvmMetricsAutoConfiguration.VirtualThreadMetricsRuntimeHintsRegistrar().registerHints(hints, + getClass().getClassLoader()); + assertThat(RuntimeHintsPredicates.reflection() + .onType(TypeReference.of(getVirtualThreadMetricsClass())) + .withMemberCategories(MemberCategory.INVOKE_PUBLIC_CONSTRUCTORS)).accepts(hints); + } + private ContextConsumer assertMetricsBeans() { return (context) -> assertThat(context).hasSingleBean(JvmGcMetrics.class) .hasSingleBean(JvmHeapPressureMetrics.class) @@ -105,6 +144,12 @@ private ContextConsumer assertMetricsBeans() { .hasSingleBean(JvmCompilationMetrics.class); } + @SuppressWarnings("unchecked") + private static Class getVirtualThreadMetricsClass() { + return (Class) ClassUtils + .resolveClassName("io.micrometer.java21.instrument.binder.jdk.VirtualThreadMetrics", null); + } + @Configuration(proxyBeanMethods = false) static class CustomJvmGcMetricsConfiguration { diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/MetricsAspectsAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/MetricsAspectsAutoConfigurationTests.java index 5371146e0f42..d01bc5c2e479 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/MetricsAspectsAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/MetricsAspectsAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -54,17 +54,6 @@ void shouldNotConfigureAspectsByDefault() { }); } - @Test - void shouldConfigureAspectsWithLegacyProperty() { - new ApplicationContextRunner().with(MetricsRun.simple()) - .withConfiguration(AutoConfigurations.of(MetricsAspectsAutoConfiguration.class)) - .withPropertyValues("micrometer.observations.annotations.enabled=true") - .run((context) -> { - assertThat(context).hasSingleBean(CountedAspect.class); - assertThat(context).hasSingleBean(TimedAspect.class); - }); - } - @Test void shouldConfigureAspects() { this.contextRunner.run((context) -> { diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/PropertiesMeterFilterTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/PropertiesMeterFilterTests.java index e1db8683bb46..1437d65e7c52 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/PropertiesMeterFilterTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/PropertiesMeterFilterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,7 +48,7 @@ class PropertiesMeterFilterTests { @Test void createWhenPropertiesIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new PropertiesMeterFilter(null)) - .withMessageContaining("Properties must not be null"); + .withMessageContaining("'properties' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesConfigAdapterTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesConfigAdapterTests.java index ba3276ac927c..cd067216b177 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesConfigAdapterTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesConfigAdapterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,7 +16,6 @@ package org.springframework.boot.actuate.autoconfigure.metrics.export.otlp; -import java.util.Collections; import java.util.Map; import java.util.concurrent.TimeUnit; @@ -30,7 +29,8 @@ import org.springframework.mock.env.MockEnvironment; import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.entry; +import static org.mockito.BDDMockito.given; +import static org.mockito.Mockito.spy; /** * Tests for {@link OtlpMetricsPropertiesConfigAdapter}. @@ -56,6 +56,20 @@ void setUp() { this.connectionDetails = new PropertiesOtlpMetricsConnectionDetails(this.properties); } + @Test + void whenPropertiesUrlIsNotSetAdapterUrlReturnsDefault() { + assertThat(this.properties.getUrl()).isNull(); + assertThat(createAdapter().url()).isEqualTo("http://localhost:4318/v1/metrics"); + } + + @Test + void whenPropertiesUrlIsNotSetThenUseOtlpConfigUrlAsFallback() { + assertThat(this.properties.getUrl()).isNull(); + OtlpMetricsPropertiesConfigAdapter adapter = spy(createAdapter()); + given(adapter.get("management.otlp.metrics.export.url")).willReturn("https://my-endpoint/v1/metrics"); + assertThat(adapter.url()).isEqualTo("https://my-endpoint/v1/metrics"); + } + @Test void whenPropertiesUrlIsSetAdapterUrlReturnsIt() { this.properties.setUrl("http://another-url:4318/v1/metrics"); @@ -74,9 +88,8 @@ void whenPropertiesAggregationTemporalityIsSetAdapterAggregationTemporalityRetur } @Test - @SuppressWarnings("removal") - void whenPropertiesResourceAttributesIsSetAdapterResourceAttributesReturnsIt() { - this.properties.setResourceAttributes(Map.of("service.name", "boot-service")); + void whenOpenTelemetryPropertiesResourceAttributesIsSetAdapterResourceAttributesReturnsIt() { + this.openTelemetryProperties.setResourceAttributes(Map.of("service.name", "boot-service")); assertThat(createAdapter().resourceAttributes()).containsEntry("service.name", "boot-service"); } @@ -131,32 +144,7 @@ void whenPropertiesBaseTimeUnitIsSetAdapterBaseTimeUnitReturnsIt() { } @Test - @SuppressWarnings("removal") - void openTelemetryPropertiesShouldOverrideOtlpPropertiesIfNotEmpty() { - this.properties.setResourceAttributes(Map.of("a", "alpha")); - this.openTelemetryProperties.setResourceAttributes(Map.of("b", "beta")); - assertThat(createAdapter().resourceAttributes()).contains(entry("b", "beta")); - assertThat(createAdapter().resourceAttributes()).doesNotContain(entry("a", "alpha")); - } - - @Test - @SuppressWarnings("removal") - void openTelemetryPropertiesShouldNotOverrideOtlpPropertiesIfEmpty() { - this.properties.setResourceAttributes(Map.of("a", "alpha")); - this.openTelemetryProperties.setResourceAttributes(Collections.emptyMap()); - assertThat(createAdapter().resourceAttributes()).contains(entry("a", "alpha")); - } - - @Test - @SuppressWarnings("removal") void serviceNameOverridesApplicationName() { - this.environment.setProperty("spring.application.name", "alpha"); - this.properties.setResourceAttributes(Map.of("service.name", "beta")); - assertThat(createAdapter().resourceAttributes()).containsEntry("service.name", "beta"); - } - - @Test - void serviceNameOverridesApplicationNameWhenUsingOtelProperties() { this.environment.setProperty("spring.application.name", "alpha"); this.openTelemetryProperties.setResourceAttributes(Map.of("service.name", "beta")); assertThat(createAdapter().resourceAttributes()).containsEntry("service.name", "beta"); @@ -174,15 +162,7 @@ void shouldUseDefaultApplicationNameIfApplicationNameIsNotSet() { } @Test - @SuppressWarnings("removal") void serviceGroupOverridesApplicationGroup() { - this.environment.setProperty("spring.application.group", "alpha"); - this.properties.setResourceAttributes(Map.of("service.group", "beta")); - assertThat(createAdapter().resourceAttributes()).containsEntry("service.group", "beta"); - } - - @Test - void serviceGroupOverridesApplicationGroupWhenUsingOtelProperties() { this.environment.setProperty("spring.application.group", "alpha"); this.openTelemetryProperties.setResourceAttributes(Map.of("service.group", "beta")); assertThat(createAdapter().resourceAttributes()).containsEntry("service.group", "beta"); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesTests.java index 36f2f85af6ec..d4938491e75b 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/otlp/OtlpMetricsPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,6 @@ void defaultValuesAreConsistent() { OtlpMetricsProperties properties = new OtlpMetricsProperties(); OtlpConfig config = OtlpConfig.DEFAULT; assertStepRegistryDefaultValues(properties, config); - assertThat(properties.getUrl()).isEqualTo(config.url()); assertThat(properties.getAggregationTemporality()).isSameAs(config.aggregationTemporality()); assertThat(properties.getHistogramFlavor()).isSameAs(config.histogramFlavor()); assertThat(properties.getMaxScale()).isEqualTo(config.maxScale()); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/DualPrometheusMetricsExportAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/DualPrometheusMetricsExportAutoConfigurationTests.java deleted file mode 100644 index f0c08e522891..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/DualPrometheusMetricsExportAutoConfigurationTests.java +++ /dev/null @@ -1,411 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus; - -import io.micrometer.core.instrument.Clock; -import io.micrometer.prometheusmetrics.PrometheusConfig; -import io.prometheus.client.CollectorRegistry; -import io.prometheus.client.exemplars.ExemplarSampler; -import io.prometheus.client.exemplars.tracer.common.SpanContextSupplier; -import io.prometheus.client.exporter.BasicAuthHttpConnectionFactory; -import io.prometheus.client.exporter.DefaultHttpConnectionFactory; -import io.prometheus.client.exporter.HttpConnectionFactory; -import io.prometheus.client.exporter.PushGateway; -import io.prometheus.metrics.model.registry.PrometheusRegistry; -import org.assertj.core.api.ThrowingConsumer; -import org.junit.jupiter.api.Test; -import org.junit.jupiter.api.extension.ExtendWith; - -import org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus.DualPrometheusMetricsExportAutoConfigurationTests.CustomSecondEndpointConfiguration.SecondPrometheusScrapeEndpoint; -import org.springframework.boot.actuate.autoconfigure.web.server.ManagementContextAutoConfiguration; -import org.springframework.boot.actuate.endpoint.web.annotation.WebEndpoint; -import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusPushGatewayManager; -import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusScrapeEndpoint; -import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusSimpleclientScrapeEndpoint; -import org.springframework.boot.autoconfigure.AutoConfigurations; -import org.springframework.boot.test.context.assertj.AssertableApplicationContext; -import org.springframework.boot.test.context.runner.ApplicationContextRunner; -import org.springframework.boot.test.context.runner.ContextConsumer; -import org.springframework.boot.test.system.CapturedOutput; -import org.springframework.boot.test.system.OutputCaptureExtension; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.context.annotation.Import; -import org.springframework.test.util.ReflectionTestUtils; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.mockito.Mockito.mock; - -/** - * Tests for {@link PrometheusSimpleclientMetricsExportAutoConfiguration} and - * {@link PrometheusMetricsExportAutoConfiguration} with both Prometheus clients on the - * classpath. - * - * @author Andy Wilkinson - * @author Stephane Nicoll - * @author Jonatan Ivanov - */ -@SuppressWarnings({ "removal", "deprecation" }) -@ExtendWith(OutputCaptureExtension.class) -class DualPrometheusMetricsExportAutoConfigurationTests { - - private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() - .withConfiguration(AutoConfigurations.of(PrometheusSimpleclientMetricsExportAutoConfiguration.class, - PrometheusMetricsExportAutoConfiguration.class)); - - @Test - void backsOffWithoutAClock() { - this.contextRunner.run((context) -> assertThat(context) - .doesNotHaveBean(io.micrometer.prometheusmetrics.PrometheusMeterRegistry.class) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusMeterRegistry.class)); - } - - @Test - void autoConfiguresItsConfigPrometheusRegistryAndMeterRegistry() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .hasSingleBean(CollectorRegistry.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusConfig.class) - .hasSingleBean(io.micrometer.prometheusmetrics.PrometheusMeterRegistry.class) - .hasSingleBean(PrometheusRegistry.class) - .hasSingleBean(io.micrometer.prometheusmetrics.PrometheusConfig.class)); - } - - @Test - void autoConfigurationCanBeDisabledWithDefaultsEnabledProperty() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .withPropertyValues("management.defaults.metrics.export.enabled=false") - .run((context) -> assertThat(context) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .doesNotHaveBean(CollectorRegistry.class) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusConfig.class) - .doesNotHaveBean(io.micrometer.prometheusmetrics.PrometheusMeterRegistry.class) - .doesNotHaveBean(PrometheusRegistry.class) - .doesNotHaveBean(io.micrometer.prometheusmetrics.PrometheusConfig.class)); - } - - @Test - void autoConfigurationCanBeDisabledWithSpecificEnabledProperty() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .withPropertyValues("management.prometheus.metrics.export.enabled=false") - .run((context) -> assertThat(context) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .doesNotHaveBean(CollectorRegistry.class) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusConfig.class) - .doesNotHaveBean(io.micrometer.prometheusmetrics.PrometheusMeterRegistry.class) - .doesNotHaveBean(PrometheusRegistry.class) - .doesNotHaveBean(io.micrometer.prometheusmetrics.PrometheusConfig.class)); - } - - @Test - void allowsCustomConfigToBeUsed() { - this.contextRunner.withUserConfiguration(CustomConfigConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .hasSingleBean(CollectorRegistry.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusConfig.class) - .hasBean("customConfig") - .hasSingleBean(PrometheusRegistry.class) - .hasSingleBean(io.micrometer.prometheusmetrics.PrometheusConfig.class) - .hasBean("otherCustomConfig")); - } - - @Test - void allowsCustomRegistryToBeUsed() { - this.contextRunner.withUserConfiguration(CustomRegistryConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .hasBean("customRegistry") - .hasSingleBean(CollectorRegistry.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusConfig.class) - .hasSingleBean(PrometheusRegistry.class) - .hasSingleBean(io.micrometer.prometheusmetrics.PrometheusConfig.class) - .hasBean("otherCustomRegistry")); - } - - @Test - void allowsCustomCollectorRegistryToBeUsed() { - this.contextRunner.withUserConfiguration(CustomCollectorRegistryConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .hasBean("customCollectorRegistry") - .hasSingleBean(CollectorRegistry.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusConfig.class) - .hasBean("customPrometheusRegistry") - .hasSingleBean(PrometheusRegistry.class) - .hasSingleBean(io.micrometer.prometheusmetrics.PrometheusConfig.class)); - } - - @Test - void autoConfiguresExemplarSamplerIfSpanContextSupplierIsPresent() { - this.contextRunner.withUserConfiguration(ExemplarsConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(SpanContextSupplier.class) - .hasSingleBean(ExemplarSampler.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class)); - } - - @Test - void allowsCustomExemplarSamplerToBeUsed() { - this.contextRunner.withUserConfiguration(ExemplarsConfiguration.class) - .withBean("customExemplarSampler", ExemplarSampler.class, () -> mock(ExemplarSampler.class)) - .run((context) -> assertThat(context).hasSingleBean(ExemplarSampler.class) - .getBean(ExemplarSampler.class) - .isSameAs(context.getBean("customExemplarSampler"))); - } - - @Test - void exemplarSamplerIsNotAutoConfiguredIfSpanContextSupplierIsMissing() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).doesNotHaveBean(SpanContextSupplier.class) - .doesNotHaveBean(ExemplarSampler.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class)); - } - - @Test - void addsScrapeEndpointToManagementContext() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withUserConfiguration(BaseConfiguration.class) - .withPropertyValues("management.endpoints.web.exposure.include=prometheus") - .run((context) -> assertThat(context).hasSingleBean(PrometheusScrapeEndpoint.class) - .doesNotHaveBean(PrometheusSimpleclientScrapeEndpoint.class)); - } - - @Test - void scrapeEndpointNotAddedToManagementContextWhenNotExposed() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).doesNotHaveBean(PrometheusSimpleclientScrapeEndpoint.class) - .doesNotHaveBean(PrometheusScrapeEndpoint.class)); - } - - @Test - void scrapeEndpointCanBeDisabled() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.endpoints.web.exposure.include=prometheus", - "management.endpoint.prometheus.enabled=false") - .withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).doesNotHaveBean(PrometheusSimpleclientScrapeEndpoint.class) - .doesNotHaveBean(PrometheusScrapeEndpoint.class)); - } - - @Test - void allowsCustomScrapeEndpointToBeUsed() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withUserConfiguration(CustomEndpointConfiguration.class) - .run((context) -> assertThat(context).hasBean("customEndpoint") - .hasSingleBean(PrometheusSimpleclientScrapeEndpoint.class)); - } - - @Test - void allowsCustomSecondScrapeEndpointToBeUsed() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withUserConfiguration(CustomSecondEndpointConfiguration.class) - .run((context) -> assertThat(context).hasBean("customSecondEndpoint") - .hasSingleBean(PrometheusSimpleclientScrapeEndpoint.class) - .hasSingleBean(SecondPrometheusScrapeEndpoint.class) - .hasSingleBean(PrometheusScrapeEndpoint.class)); - } - - @Test - void pushGatewayIsNotConfiguredWhenEnabledFlagIsNotSet() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).doesNotHaveBean(PrometheusPushGatewayManager.class)); - } - - @Test - void withPushGatewayEnabled(CapturedOutput output) { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true") - .withUserConfiguration(BaseConfiguration.class) - .run((context) -> { - assertThat(output).doesNotContain("Invalid PushGateway base url"); - hasGatewayURL(context, "http://localhost:9091/metrics/"); - }); - } - - @Test - void withPushGatewayNoBasicAuth() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true") - .withUserConfiguration(BaseConfiguration.class) - .run(hasHttpConnectionFactory((httpConnectionFactory) -> assertThat(httpConnectionFactory) - .isInstanceOf(DefaultHttpConnectionFactory.class))); - } - - @Test - void withCustomPushGatewayURL() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", - "management.prometheus.metrics.export.pushgateway.base-url=https://example.com:8080") - .withUserConfiguration(BaseConfiguration.class) - .run((context) -> hasGatewayURL(context, "https://example.com:8080/metrics/")); - } - - @Test - void withPushGatewayBasicAuth() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", - "management.prometheus.metrics.export.pushgateway.username=admin", - "management.prometheus.metrics.export.pushgateway.password=secret") - .withUserConfiguration(BaseConfiguration.class) - .run(hasHttpConnectionFactory((httpConnectionFactory) -> assertThat(httpConnectionFactory) - .isInstanceOf(BasicAuthHttpConnectionFactory.class))); - } - - private void hasGatewayURL(AssertableApplicationContext context, String url) { - assertThat(getPushGateway(context)).hasFieldOrPropertyWithValue("gatewayBaseURL", url); - } - - private ContextConsumer hasHttpConnectionFactory( - ThrowingConsumer httpConnectionFactory) { - return (context) -> { - PushGateway pushGateway = getPushGateway(context); - httpConnectionFactory - .accept((HttpConnectionFactory) ReflectionTestUtils.getField(pushGateway, "connectionFactory")); - }; - } - - private PushGateway getPushGateway(AssertableApplicationContext context) { - assertThat(context).hasSingleBean(PrometheusPushGatewayManager.class); - PrometheusPushGatewayManager gatewayManager = context.getBean(PrometheusPushGatewayManager.class); - return (PushGateway) ReflectionTestUtils.getField(gatewayManager, "pushGateway"); - } - - @Configuration(proxyBeanMethods = false) - static class BaseConfiguration { - - @Bean - Clock clock() { - return Clock.SYSTEM; - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class CustomConfigConfiguration { - - @Bean - io.micrometer.prometheus.PrometheusConfig customConfig() { - return (key) -> null; - } - - @Bean - io.micrometer.prometheusmetrics.PrometheusConfig otherCustomConfig() { - return (key) -> null; - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class CustomRegistryConfiguration { - - @Bean - io.micrometer.prometheus.PrometheusMeterRegistry customRegistry( - io.micrometer.prometheus.PrometheusConfig config, CollectorRegistry collectorRegistry, Clock clock) { - return new io.micrometer.prometheus.PrometheusMeterRegistry(config, collectorRegistry, clock); - } - - @Bean - io.micrometer.prometheusmetrics.PrometheusMeterRegistry otherCustomRegistry( - io.micrometer.prometheusmetrics.PrometheusConfig config, PrometheusRegistry prometheusRegistry, - Clock clock) { - return new io.micrometer.prometheusmetrics.PrometheusMeterRegistry(config, prometheusRegistry, clock); - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class CustomCollectorRegistryConfiguration { - - @Bean - CollectorRegistry customCollectorRegistry() { - return new CollectorRegistry(); - } - - @Bean - PrometheusRegistry customPrometheusRegistry() { - return new PrometheusRegistry(); - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class CustomEndpointConfiguration { - - @Bean - PrometheusSimpleclientScrapeEndpoint customEndpoint(CollectorRegistry collectorRegistry) { - return new PrometheusSimpleclientScrapeEndpoint(collectorRegistry); - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class CustomSecondEndpointConfiguration { - - @Bean - PrometheusScrapeEndpoint prometheusScrapeEndpoint(PrometheusRegistry prometheusRegistry, - PrometheusConfig prometheusConfig) { - return new PrometheusScrapeEndpoint(prometheusRegistry, prometheusConfig.prometheusProperties()); - } - - @Bean - SecondPrometheusScrapeEndpoint customSecondEndpoint(CollectorRegistry collectorRegistry) { - return new SecondPrometheusScrapeEndpoint(collectorRegistry); - } - - @WebEndpoint(id = "prometheussc") - static class SecondPrometheusScrapeEndpoint extends PrometheusSimpleclientScrapeEndpoint { - - SecondPrometheusScrapeEndpoint(CollectorRegistry collectorRegistry) { - super(collectorRegistry); - } - - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class ExemplarsConfiguration { - - @Bean - SpanContextSupplier spanContextSupplier() { - return new SpanContextSupplier() { - - @Override - public String getTraceId() { - return null; - } - - @Override - public String getSpanId() { - return null; - } - - @Override - public boolean isSampled() { - return false; - } - - }; - } - - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusMetricsExportAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusMetricsExportAutoConfigurationTests.java index 1c26b2ec7135..9c8013281d99 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusMetricsExportAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusMetricsExportAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,22 +16,34 @@ package org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus; +import java.net.MalformedURLException; +import java.net.URI; + import io.micrometer.core.instrument.Clock; import io.micrometer.prometheusmetrics.PrometheusConfig; import io.micrometer.prometheusmetrics.PrometheusMeterRegistry; +import io.prometheus.metrics.exporter.pushgateway.DefaultHttpConnectionFactory; +import io.prometheus.metrics.exporter.pushgateway.PushGateway; +import io.prometheus.metrics.expositionformats.PrometheusTextFormatWriter; import io.prometheus.metrics.model.registry.PrometheusRegistry; import io.prometheus.metrics.tracer.common.SpanContext; +import org.assertj.core.api.InstanceOfAssertFactories; import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; import org.springframework.boot.actuate.autoconfigure.web.server.ManagementContextAutoConfiguration; import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusPushGatewayManager; import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusScrapeEndpoint; import org.springframework.boot.autoconfigure.AutoConfigurations; -import org.springframework.boot.test.context.FilteredClassLoader; +import org.springframework.boot.context.properties.source.MutuallyExclusiveConfigurationPropertiesException; +import org.springframework.boot.test.context.assertj.AssertableApplicationContext; import org.springframework.boot.test.context.runner.ApplicationContextRunner; +import org.springframework.boot.test.system.CapturedOutput; +import org.springframework.boot.test.system.OutputCaptureExtension; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Import; +import org.springframework.test.util.ReflectionTestUtils; import static org.assertj.core.api.Assertions.assertThat; @@ -45,7 +57,6 @@ class PrometheusMetricsExportAutoConfigurationTests { private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() - .withClassLoader(new FilteredClassLoader("io.micrometer.prometheus.", "io.prometheus.client")) .withConfiguration(AutoConfigurations.of(PrometheusMetricsExportAutoConfiguration.class)); @Test @@ -158,6 +169,106 @@ void pushGatewayIsNotConfiguredWhenEnabledFlagIsNotSet() { .run((context) -> assertThat(context).doesNotHaveBean(PrometheusPushGatewayManager.class)); } + @Test + @ExtendWith(OutputCaptureExtension.class) + void withPushGatewayEnabled(CapturedOutput output) { + this.contextRunner.withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true") + .withUserConfiguration(BaseConfiguration.class) + .run((context) -> { + assertThat(output).doesNotContain("Invalid PushGateway base url"); + hasGatewayUrl(context, "http://localhost:9091/metrics/job/spring"); + assertThat(getPushGateway(context)).extracting("connectionFactory") + .isInstanceOf(DefaultHttpConnectionFactory.class); + }); + } + + @Test + void withPushGatewayDisabled() { + this.contextRunner.withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=false") + .withUserConfiguration(BaseConfiguration.class) + .run((context) -> assertThat(context).doesNotHaveBean(PrometheusPushGatewayManager.class)); + } + + @Test + void withCustomPushGatewayAddress() { + this.contextRunner + .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", + "management.prometheus.metrics.export.pushgateway.address=localhost:8080") + .withUserConfiguration(BaseConfiguration.class) + .run((context) -> hasGatewayUrl(context, "http://localhost:8080/metrics/job/spring")); + } + + @Test + void withCustomScheme() { + this.contextRunner + .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", + "management.prometheus.metrics.export.pushgateway.scheme=https") + .withUserConfiguration(BaseConfiguration.class) + .run((context) -> hasGatewayUrl(context, "https://localhost:9091/metrics/job/spring")); + } + + @Test + void withCustomFormat() { + this.contextRunner + .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", + "management.prometheus.metrics.export.pushgateway.format=text") + .withUserConfiguration(BaseConfiguration.class) + .run((context) -> assertThat(getPushGateway(context)).extracting("writer") + .isInstanceOf(PrometheusTextFormatWriter.class)); + } + + @Test + void withPushGatewayBasicAuth() { + this.contextRunner + .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", + "management.prometheus.metrics.export.pushgateway.username=admin", + "management.prometheus.metrics.export.pushgateway.password=secret") + .withUserConfiguration(BaseConfiguration.class) + .run((context) -> assertThat(getPushGateway(context)) + .extracting("requestHeaders", InstanceOfAssertFactories.map(String.class, String.class)) + .satisfies((headers) -> assertThat(headers.get("Authorization")).startsWith("Basic "))); + + } + + @Test + void withPushGatewayBearerToken() { + this.contextRunner + .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", + "management.prometheus.metrics.export.pushgateway.token=a1b2c3d4") + .withUserConfiguration(BaseConfiguration.class) + .run((context) -> assertThat(getPushGateway(context)) + .extracting("requestHeaders", InstanceOfAssertFactories.map(String.class, String.class)) + .satisfies((headers) -> assertThat(headers.get("Authorization")).startsWith("Bearer "))); + } + + @Test + void failsFastWithBothBearerAndBasicAuthentication() { + this.contextRunner + .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", + "management.prometheus.metrics.export.pushgateway.username=alice", + "management.prometheus.metrics.export.pushgateway.token=a1b2c3d4") + .withUserConfiguration(BaseConfiguration.class) + .run((context) -> assertThat(context).getFailure() + .hasRootCauseInstanceOf(MutuallyExclusiveConfigurationPropertiesException.class) + .hasMessageContainingAll("management.prometheus.metrics.export.pushgateway.username", + "management.prometheus.metrics.export.pushgateway.token")); + } + + private void hasGatewayUrl(AssertableApplicationContext context, String url) { + try { + assertThat(getPushGateway(context)).hasFieldOrPropertyWithValue("url", URI.create(url).toURL()); + } + catch (MalformedURLException ex) { + throw new RuntimeException(ex); + } + } + + private PushGateway getPushGateway(AssertableApplicationContext context) { + assertThat(context).hasSingleBean(PrometheusPushGatewayManager.class); + PrometheusPushGatewayManager gatewayManager = context.getBean(PrometheusPushGatewayManager.class); + return (PushGateway) ReflectionTestUtils.getField(gatewayManager, "pushGateway"); + } + @Configuration(proxyBeanMethods = false) static class BaseConfiguration { diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusPropertiesTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusPropertiesTests.java index cfdd0a4188c8..225fc5626f37 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusPropertiesTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,15 +35,4 @@ void defaultValuesAreConsistent() { assertThat(properties.getStep()).isEqualTo(config.step()); } - @SuppressWarnings("deprecation") - @Test - void defaultValuesAreConsistentWithSimpleclient() { - PrometheusProperties properties = new PrometheusProperties(); - io.micrometer.prometheus.PrometheusConfig config = io.micrometer.prometheus.PrometheusConfig.DEFAULT; - assertThat(properties.isDescriptions()).isEqualTo(config.descriptions()); - assertThat(PrometheusSimpleclientPropertiesConfigAdapter.mapToMicrometerHistogramFlavor(properties)) - .isEqualTo(config.histogramFlavor()); - assertThat(properties.getStep()).isEqualTo(config.step()); - } - } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusScrapeEndpointDocumentationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusScrapeEndpointDocumentationTests.java index 79124e8cdca7..8a82b32fee45 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusScrapeEndpointDocumentationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusScrapeEndpointDocumentationTests.java @@ -21,7 +21,7 @@ import io.micrometer.core.instrument.Clock; import io.micrometer.core.instrument.binder.jvm.JvmMemoryMetrics; import io.micrometer.prometheusmetrics.PrometheusMeterRegistry; -import io.prometheus.client.exporter.common.TextFormat; +import io.prometheus.metrics.expositionformats.OpenMetricsTextFormatWriter; import io.prometheus.metrics.model.registry.PrometheusRegistry; import org.junit.jupiter.api.Test; @@ -50,7 +50,7 @@ void prometheus() { @Test void prometheusOpenmetrics() { - assertThat(this.mvc.get().uri("/actuator/prometheus").accept(TextFormat.CONTENT_TYPE_OPENMETRICS_100)) + assertThat(this.mvc.get().uri("/actuator/prometheus").accept(OpenMetricsTextFormatWriter.CONTENT_TYPE)) .satisfies((result) -> { assertThat(result).hasStatusOk() .headers() diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientMetricsExportAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientMetricsExportAutoConfigurationTests.java deleted file mode 100644 index b1778b76f6f5..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientMetricsExportAutoConfigurationTests.java +++ /dev/null @@ -1,330 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus; - -import io.micrometer.core.instrument.Clock; -import io.prometheus.client.CollectorRegistry; -import io.prometheus.client.exemplars.ExemplarSampler; -import io.prometheus.client.exemplars.tracer.common.SpanContextSupplier; -import io.prometheus.client.exporter.BasicAuthHttpConnectionFactory; -import io.prometheus.client.exporter.DefaultHttpConnectionFactory; -import io.prometheus.client.exporter.HttpConnectionFactory; -import io.prometheus.client.exporter.PushGateway; -import org.assertj.core.api.ThrowingConsumer; -import org.junit.jupiter.api.Test; -import org.junit.jupiter.api.extension.ExtendWith; - -import org.springframework.boot.actuate.autoconfigure.web.server.ManagementContextAutoConfiguration; -import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusPushGatewayManager; -import org.springframework.boot.actuate.metrics.export.prometheus.PrometheusSimpleclientScrapeEndpoint; -import org.springframework.boot.autoconfigure.AutoConfigurations; -import org.springframework.boot.test.context.FilteredClassLoader; -import org.springframework.boot.test.context.assertj.AssertableApplicationContext; -import org.springframework.boot.test.context.runner.ApplicationContextRunner; -import org.springframework.boot.test.context.runner.ContextConsumer; -import org.springframework.boot.test.system.CapturedOutput; -import org.springframework.boot.test.system.OutputCaptureExtension; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.context.annotation.Import; -import org.springframework.test.util.ReflectionTestUtils; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.mockito.Mockito.mock; - -/** - * Tests for {@link PrometheusSimpleclientMetricsExportAutoConfiguration}. - * - * @author Andy Wilkinson - * @author Stephane Nicoll - * @author Jonatan Ivanov - */ -@SuppressWarnings({ "removal", "deprecation" }) -@ExtendWith(OutputCaptureExtension.class) -class PrometheusSimpleclientMetricsExportAutoConfigurationTests { - - private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() - .withClassLoader(new FilteredClassLoader("io.micrometer.prometheusmetrics.", "io.prometheus.metrics")) - .withConfiguration(AutoConfigurations.of(PrometheusSimpleclientMetricsExportAutoConfiguration.class)); - - @Test - void backsOffWithoutAClock() { - this.contextRunner.run((context) -> assertThat(context) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusMeterRegistry.class)); - } - - @Test - void autoConfiguresItsConfigCollectorRegistryAndMeterRegistry() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .hasSingleBean(CollectorRegistry.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusConfig.class)); - } - - @Test - void autoConfigurationCanBeDisabledWithDefaultsEnabledProperty() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .withPropertyValues("management.defaults.metrics.export.enabled=false") - .run((context) -> assertThat(context) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .doesNotHaveBean(CollectorRegistry.class) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusConfig.class)); - } - - @Test - void autoConfigurationCanBeDisabledWithSpecificEnabledProperty() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .withPropertyValues("management.prometheus.metrics.export.enabled=false") - .run((context) -> assertThat(context) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .doesNotHaveBean(CollectorRegistry.class) - .doesNotHaveBean(io.micrometer.prometheus.PrometheusConfig.class)); - } - - @Test - void allowsCustomConfigToBeUsed() { - this.contextRunner.withUserConfiguration(CustomConfigConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .hasSingleBean(CollectorRegistry.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusConfig.class) - .hasBean("customConfig")); - } - - @Test - void allowsCustomRegistryToBeUsed() { - this.contextRunner.withUserConfiguration(CustomRegistryConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .hasBean("customRegistry") - .hasSingleBean(CollectorRegistry.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusConfig.class)); - } - - @Test - void allowsCustomCollectorRegistryToBeUsed() { - this.contextRunner.withUserConfiguration(CustomCollectorRegistryConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class) - .hasBean("customCollectorRegistry") - .hasSingleBean(CollectorRegistry.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusConfig.class)); - } - - @Test - void autoConfiguresExemplarSamplerIfSpanContextSupplierIsPresent() { - this.contextRunner.withUserConfiguration(ExemplarsConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(SpanContextSupplier.class) - .hasSingleBean(ExemplarSampler.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class)); - } - - @Test - void allowsCustomExemplarSamplerToBeUsed() { - this.contextRunner.withUserConfiguration(ExemplarsConfiguration.class) - .withBean("customExemplarSampler", ExemplarSampler.class, () -> mock(ExemplarSampler.class)) - .run((context) -> assertThat(context).hasSingleBean(ExemplarSampler.class) - .getBean(ExemplarSampler.class) - .isSameAs(context.getBean("customExemplarSampler"))); - } - - @Test - void exemplarSamplerIsNotAutoConfiguredIfSpanContextSupplierIsMissing() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).doesNotHaveBean(SpanContextSupplier.class) - .doesNotHaveBean(ExemplarSampler.class) - .hasSingleBean(io.micrometer.prometheus.PrometheusMeterRegistry.class)); - } - - @Test - void addsScrapeEndpointToManagementContext() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withUserConfiguration(BaseConfiguration.class) - .withPropertyValues("management.endpoints.web.exposure.include=prometheus") - .run((context) -> assertThat(context).hasSingleBean(PrometheusSimpleclientScrapeEndpoint.class)); - } - - @Test - void scrapeEndpointNotAddedToManagementContextWhenNotExposed() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).doesNotHaveBean(PrometheusSimpleclientScrapeEndpoint.class)); - } - - @Test - void scrapeEndpointCanBeDisabled() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.endpoints.web.exposure.include=prometheus", - "management.endpoint.prometheus.enabled=false") - .withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).doesNotHaveBean(PrometheusSimpleclientScrapeEndpoint.class)); - } - - @Test - void allowsCustomScrapeEndpointToBeUsed() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withUserConfiguration(CustomEndpointConfiguration.class) - .run((context) -> assertThat(context).hasBean("customEndpoint") - .hasSingleBean(PrometheusSimpleclientScrapeEndpoint.class)); - } - - @Test - void pushGatewayIsNotConfiguredWhenEnabledFlagIsNotSet() { - this.contextRunner.withUserConfiguration(BaseConfiguration.class) - .run((context) -> assertThat(context).doesNotHaveBean(PrometheusPushGatewayManager.class)); - } - - @Test - void withPushGatewayEnabled(CapturedOutput output) { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true") - .withUserConfiguration(BaseConfiguration.class) - .run((context) -> { - assertThat(output).doesNotContain("Invalid PushGateway base url"); - hasGatewayURL(context, "http://localhost:9091/metrics/"); - }); - } - - @Test - void withPushGatewayNoBasicAuth() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true") - .withUserConfiguration(BaseConfiguration.class) - .run(hasHttpConnectionFactory((httpConnectionFactory) -> assertThat(httpConnectionFactory) - .isInstanceOf(DefaultHttpConnectionFactory.class))); - } - - @Test - void withCustomPushGatewayURL() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", - "management.prometheus.metrics.export.pushgateway.base-url=https://example.com:8080") - .withUserConfiguration(BaseConfiguration.class) - .run((context) -> hasGatewayURL(context, "https://example.com:8080/metrics/")); - } - - @Test - void withPushGatewayBasicAuth() { - this.contextRunner.withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class)) - .withPropertyValues("management.prometheus.metrics.export.pushgateway.enabled=true", - "management.prometheus.metrics.export.pushgateway.username=admin", - "management.prometheus.metrics.export.pushgateway.password=secret") - .withUserConfiguration(BaseConfiguration.class) - .run(hasHttpConnectionFactory((httpConnectionFactory) -> assertThat(httpConnectionFactory) - .isInstanceOf(BasicAuthHttpConnectionFactory.class))); - } - - private void hasGatewayURL(AssertableApplicationContext context, String url) { - assertThat(getPushGateway(context)).hasFieldOrPropertyWithValue("gatewayBaseURL", url); - } - - private ContextConsumer hasHttpConnectionFactory( - ThrowingConsumer httpConnectionFactory) { - return (context) -> { - PushGateway pushGateway = getPushGateway(context); - httpConnectionFactory - .accept((HttpConnectionFactory) ReflectionTestUtils.getField(pushGateway, "connectionFactory")); - }; - } - - private PushGateway getPushGateway(AssertableApplicationContext context) { - assertThat(context).hasSingleBean(PrometheusPushGatewayManager.class); - PrometheusPushGatewayManager gatewayManager = context.getBean(PrometheusPushGatewayManager.class); - return (PushGateway) ReflectionTestUtils.getField(gatewayManager, "pushGateway"); - } - - @Configuration(proxyBeanMethods = false) - static class BaseConfiguration { - - @Bean - Clock clock() { - return Clock.SYSTEM; - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class CustomConfigConfiguration { - - @Bean - io.micrometer.prometheus.PrometheusConfig customConfig() { - return (key) -> null; - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class CustomRegistryConfiguration { - - @Bean - io.micrometer.prometheus.PrometheusMeterRegistry customRegistry( - io.micrometer.prometheus.PrometheusConfig config, CollectorRegistry collectorRegistry, Clock clock) { - return new io.micrometer.prometheus.PrometheusMeterRegistry(config, collectorRegistry, clock); - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class CustomCollectorRegistryConfiguration { - - @Bean - CollectorRegistry customCollectorRegistry() { - return new CollectorRegistry(); - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class CustomEndpointConfiguration { - - @Bean - PrometheusSimpleclientScrapeEndpoint customEndpoint(CollectorRegistry collectorRegistry) { - return new PrometheusSimpleclientScrapeEndpoint(collectorRegistry); - } - - } - - @Configuration(proxyBeanMethods = false) - @Import(BaseConfiguration.class) - static class ExemplarsConfiguration { - - @Bean - SpanContextSupplier spanContextSupplier() { - return new SpanContextSupplier() { - - @Override - public String getTraceId() { - return null; - } - - @Override - public String getSpanId() { - return null; - } - - @Override - public boolean isSampled() { - return false; - } - - }; - } - - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientPropertiesConfigAdapterTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientPropertiesConfigAdapterTests.java deleted file mode 100644 index cfd4701d6d71..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientPropertiesConfigAdapterTests.java +++ /dev/null @@ -1,64 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus; - -import java.time.Duration; - -import org.junit.jupiter.api.Test; - -import org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus.PrometheusProperties.HistogramFlavor; -import org.springframework.boot.actuate.autoconfigure.metrics.export.properties.AbstractPropertiesConfigAdapterTests; - -import static org.assertj.core.api.Assertions.assertThat; - -/** - * Tests for {@link PrometheusSimpleclientPropertiesConfigAdapter}. - * - * @author Mirko Sobeck - */ -@SuppressWarnings({ "deprecation", "removal" }) -class PrometheusSimpleclientPropertiesConfigAdapterTests extends - AbstractPropertiesConfigAdapterTests { - - PrometheusSimpleclientPropertiesConfigAdapterTests() { - super(PrometheusSimpleclientPropertiesConfigAdapter.class); - } - - @Test - void whenPropertiesDescriptionsIsSetAdapterDescriptionsReturnsIt() { - PrometheusProperties properties = new PrometheusProperties(); - properties.setDescriptions(false); - assertThat(new PrometheusSimpleclientPropertiesConfigAdapter(properties).descriptions()).isFalse(); - } - - @Test - void whenPropertiesHistogramFlavorIsSetAdapterHistogramFlavorReturnsIt() { - PrometheusProperties properties = new PrometheusProperties(); - properties.setHistogramFlavor(HistogramFlavor.VictoriaMetrics); - assertThat(new PrometheusSimpleclientPropertiesConfigAdapter(properties).histogramFlavor()) - .isEqualTo(io.micrometer.prometheus.HistogramFlavor.VictoriaMetrics); - } - - @Test - void whenPropertiesStepIsSetAdapterStepReturnsIt() { - PrometheusProperties properties = new PrometheusProperties(); - properties.setStep(Duration.ofSeconds(30)); - assertThat(new PrometheusSimpleclientPropertiesConfigAdapter(properties).step()) - .isEqualTo(Duration.ofSeconds(30)); - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientScrapeEndpointDocumentationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientScrapeEndpointDocumentationTests.java deleted file mode 100644 index 195fde34c69a..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/prometheus/PrometheusSimpleclientScrapeEndpointDocumentationTests.java +++ /dev/null @@ -1,89 +0,0 @@ -/* - * Copyright 2012-2025 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus; - -import io.micrometer.core.instrument.Clock; -import io.micrometer.core.instrument.binder.jvm.JvmMemoryMetrics; -import io.prometheus.client.CollectorRegistry; -import io.prometheus.client.exporter.common.TextFormat; -import org.junit.jupiter.api.Test; - -import org.springframework.boot.actuate.autoconfigure.endpoint.web.documentation.MockMvcEndpointDocumentationTests; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.springframework.restdocs.mockmvc.MockMvcRestDocumentation.document; -import static org.springframework.restdocs.request.RequestDocumentation.parameterWithName; -import static org.springframework.restdocs.request.RequestDocumentation.queryParameters; - -/** - * Tests for generating documentation describing the - * {@link org.springframework.boot.actuate.metrics.export.prometheus.PrometheusSimpleclientScrapeEndpoint}. - * - * @author Andy Wilkinson - * @author Johnny Lim - */ -class PrometheusSimpleclientScrapeEndpointDocumentationTests extends MockMvcEndpointDocumentationTests { - - @Test - void prometheus() { - assertThat(this.mvc.get().uri("/actuator/prometheus")).hasStatusOk() - .apply(document("prometheus-simpleclient/all")); - } - - @Test - void prometheusOpenmetrics() { - assertThat(this.mvc.get().uri("/actuator/prometheus").accept(TextFormat.CONTENT_TYPE_OPENMETRICS_100)) - .satisfies((result) -> { - assertThat(result).hasStatusOk() - .headers() - .hasValue("Content-Type", "application/openmetrics-text;version=1.0.0;charset=utf-8"); - assertThat(result).apply(document("prometheus-simpleclient/openmetrics")); - }); - - } - - @Test - void filteredPrometheus() { - assertThat(this.mvc.get() - .uri("/actuator/prometheus") - .param("includedNames", "jvm_memory_used_bytes,jvm_memory_committed_bytes")) - .hasStatusOk() - .apply(document("prometheus-simpleclient/names", - queryParameters(parameterWithName("includedNames") - .description("Restricts the samples to those that match the names. Optional.") - .optional()))); - } - - @Configuration(proxyBeanMethods = false) - static class TestConfiguration { - - @Bean - @SuppressWarnings({ "removal", "deprecation" }) - org.springframework.boot.actuate.metrics.export.prometheus.PrometheusSimpleclientScrapeEndpoint endpoint() { - CollectorRegistry collectorRegistry = new CollectorRegistry(true); - io.micrometer.prometheus.PrometheusMeterRegistry meterRegistry = new io.micrometer.prometheus.PrometheusMeterRegistry( - (key) -> null, collectorRegistry, Clock.SYSTEM); - new JvmMemoryMetrics().bindTo(meterRegistry); - return new org.springframework.boot.actuate.metrics.export.prometheus.PrometheusSimpleclientScrapeEndpoint( - collectorRegistry); - } - - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxMetricsExportAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxMetricsExportAutoConfigurationTests.java index d3db91eacb10..591cb645f48d 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxMetricsExportAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxMetricsExportAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,7 +33,10 @@ * Tests for {@link SignalFxMetricsExportAutoConfiguration}. * * @author Andy Wilkinson + * @deprecated since 3.5.0 for removal in 4.0.0 */ +@SuppressWarnings("removal") +@Deprecated(since = "3.5.0", forRemoval = true) class SignalFxMetricsExportAutoConfigurationTests { private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesConfigAdapterTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesConfigAdapterTests.java index d664a342148a..6383f04f9a26 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesConfigAdapterTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesConfigAdapterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,7 +19,6 @@ import org.junit.jupiter.api.Test; import org.springframework.boot.actuate.autoconfigure.metrics.export.properties.StepRegistryPropertiesConfigAdapterTests; -import org.springframework.boot.actuate.autoconfigure.metrics.export.signalfx.SignalFxProperties.HistogramType; import static org.assertj.core.api.Assertions.assertThat; @@ -27,7 +26,10 @@ * Tests for {@link SignalFxPropertiesConfigAdapter}. * * @author Mirko Sobeck + * @deprecated since 3.5.0 for removal in 4.0.0 */ +@SuppressWarnings("removal") +@Deprecated(since = "3.5.0", forRemoval = true) class SignalFxPropertiesConfigAdapterTests extends StepRegistryPropertiesConfigAdapterTests { @@ -70,7 +72,7 @@ void whenPropertiesSourceIsSetAdapterSourceReturnsIt() { @Test void whenPropertiesPublishHistogramTypeIsCumulativeAdapterPublishCumulativeHistogramReturnsIt() { SignalFxProperties properties = createProperties(); - properties.setPublishedHistogramType(HistogramType.CUMULATIVE); + properties.setPublishedHistogramType(SignalFxProperties.HistogramType.CUMULATIVE); assertThat(createConfigAdapter(properties).publishCumulativeHistogram()).isTrue(); assertThat(createConfigAdapter(properties).publishDeltaHistogram()).isFalse(); } @@ -78,7 +80,7 @@ void whenPropertiesPublishHistogramTypeIsCumulativeAdapterPublishCumulativeHisto @Test void whenPropertiesPublishHistogramTypeIsDeltaAdapterPublishDeltaHistogramReturnsIt() { SignalFxProperties properties = createProperties(); - properties.setPublishedHistogramType(HistogramType.DELTA); + properties.setPublishedHistogramType(SignalFxProperties.HistogramType.DELTA); assertThat(createConfigAdapter(properties).publishDeltaHistogram()).isTrue(); assertThat(createConfigAdapter(properties).publishCumulativeHistogram()).isFalse(); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesTests.java index ff7b6bf380e8..0fb6ffbd1543 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/export/signalfx/SignalFxPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,7 +20,6 @@ import org.junit.jupiter.api.Test; import org.springframework.boot.actuate.autoconfigure.metrics.export.properties.StepRegistryPropertiesTests; -import org.springframework.boot.actuate.autoconfigure.metrics.export.signalfx.SignalFxProperties.HistogramType; import static org.assertj.core.api.Assertions.assertThat; @@ -28,7 +27,10 @@ * Tests for {@link SignalFxProperties}. * * @author Stephane Nicoll + * @deprecated since 3.5.0 for removal in 4.0.0 */ +@SuppressWarnings("removal") +@Deprecated(since = "3.5.0", forRemoval = true) class SignalFxPropertiesTests extends StepRegistryPropertiesTests { @Test @@ -43,7 +45,7 @@ void defaultValuesAreConsistent() { // histogram type should be published. assertThat(config.publishCumulativeHistogram()).isFalse(); assertThat(config.publishDeltaHistogram()).isFalse(); - assertThat(properties.getPublishedHistogramType()).isEqualTo(HistogramType.DEFAULT); + assertThat(properties.getPublishedHistogramType()).isEqualTo(SignalFxProperties.HistogramType.DEFAULT); } } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/test/MetricsRun.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/test/MetricsRun.java index 95fef97721bf..729504530d42 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/test/MetricsRun.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/metrics/test/MetricsRun.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,6 @@ import org.springframework.boot.actuate.autoconfigure.metrics.export.newrelic.NewRelicMetricsExportAutoConfiguration; import org.springframework.boot.actuate.autoconfigure.metrics.export.otlp.OtlpMetricsExportAutoConfiguration; import org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus.PrometheusMetricsExportAutoConfiguration; -import org.springframework.boot.actuate.autoconfigure.metrics.export.signalfx.SignalFxMetricsExportAutoConfiguration; import org.springframework.boot.actuate.autoconfigure.metrics.export.simple.SimpleMetricsExportAutoConfiguration; import org.springframework.boot.actuate.autoconfigure.metrics.export.statsd.StatsdMetricsExportAutoConfiguration; import org.springframework.boot.autoconfigure.AutoConfigurations; @@ -47,6 +46,7 @@ * @author Jon Schneider * @author Phillip Webb */ +@SuppressWarnings("removal") public final class MetricsRun { private static final Set> EXPORT_AUTO_CONFIGURATIONS; @@ -63,7 +63,8 @@ public final class MetricsRun { implementations.add(OtlpMetricsExportAutoConfiguration.class); implementations.add(PrometheusMetricsExportAutoConfiguration.class); implementations.add(SimpleMetricsExportAutoConfiguration.class); - implementations.add(SignalFxMetricsExportAutoConfiguration.class); + implementations.add( + org.springframework.boot.actuate.autoconfigure.metrics.export.signalfx.SignalFxMetricsExportAutoConfiguration.class); implementations.add(StatsdMetricsExportAutoConfiguration.class); EXPORT_AUTO_CONFIGURATIONS = Collections.unmodifiableSet(implementations); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/observation/web/servlet/WebMvcObservationAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/observation/web/servlet/WebMvcObservationAutoConfigurationTests.java index f9ba22dedae7..93ef6d904607 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/observation/web/servlet/WebMvcObservationAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/observation/web/servlet/WebMvcObservationAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryAutoConfigurationTests.java index 14ab7b3c6380..161d3f37f095 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -106,6 +106,23 @@ void shouldNotApplySpringApplicationGroupIfNotSet() { }); } + @Test + void shouldApplyServiceNamespaceIfApplicationGroupIsSet() { + this.runner.withPropertyValues("spring.application.group=my-group").run((context) -> { + Resource resource = context.getBean(Resource.class); + assertThat(resource.getAttributes().asMap()).containsEntry(AttributeKey.stringKey("service.namespace"), + "my-group"); + }); + } + + @Test + void shouldNotApplyServiceNamespaceIfApplicationGroupIsNotSet() { + this.runner.run(((context) -> { + Resource resource = context.getBean(Resource.class); + assertThat(resource.getAttributes().asMap()).doesNotContainKey(AttributeKey.stringKey("service.namespace")); + })); + } + @Test void shouldFallbackToDefaultApplicationNameIfSpringApplicationNameIsNotSet() { this.runner.run((context) -> { @@ -156,7 +173,7 @@ void shouldRegisterSdkMeterProviderIfAvailable() { } @Configuration(proxyBeanMethods = false) - private static final class UserConfiguration { + static class UserConfiguration { @Bean OpenTelemetry customOpenTelemetry() { diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryResourceAttributesTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryResourceAttributesTests.java new file mode 100644 index 000000000000..9481f94e7ae5 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/opentelemetry/OpenTelemetryResourceAttributesTests.java @@ -0,0 +1,258 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.opentelemetry; + +import java.util.LinkedHashMap; +import java.util.Map; +import java.util.function.Function; +import java.util.stream.IntStream; + +import io.opentelemetry.api.internal.PercentEscaper; +import org.assertj.core.api.InstanceOfAssertFactories; +import org.junit.jupiter.api.Test; + +import org.springframework.mock.env.MockEnvironment; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; + +/** + * Tests for {@link OpenTelemetryResourceAttributes}. + * + * @author Dmytro Nosan + */ +class OpenTelemetryResourceAttributesTests { + + private final MockEnvironment environment = new MockEnvironment(); + + private final Map environmentVariables = new LinkedHashMap<>(); + + private final Map resourceAttributes = new LinkedHashMap<>(); + + @Test + void otelServiceNameShouldTakePrecedenceOverOtelResourceAttributes() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "service.name=ignored"); + this.environmentVariables.put("OTEL_SERVICE_NAME", "otel-service"); + assertThat(getAttributes()).hasSize(1).containsEntry("service.name", "otel-service"); + } + + @Test + void otelServiceNameWhenEmptyShouldTakePrecedenceOverOtelResourceAttributes() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "service.name=ignored"); + this.environmentVariables.put("OTEL_SERVICE_NAME", ""); + assertThat(getAttributes()).hasSize(1).containsEntry("service.name", ""); + } + + @Test + void otelResourceAttributes() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", + ", ,,key1=value1,key2= value2, key3=value3,key4=,=value5,key6,=,key7=%20spring+boot%20,key8=ś"); + assertThat(getAttributes()).hasSize(7) + .containsEntry("key1", "value1") + .containsEntry("key2", "value2") + .containsEntry("key3", "value3") + .containsEntry("key4", "") + .containsEntry("key7", " spring+boot ") + .containsEntry("key8", "ś") + .containsEntry("service.name", "unknown_service"); + } + + @Test + void resourceAttributesShouldBeMergedWithEnvironmentVariablesAndTakePrecedence() { + this.resourceAttributes.put("service.group", "custom-group"); + this.resourceAttributes.put("key2", ""); + this.environmentVariables.put("OTEL_SERVICE_NAME", "custom-service"); + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "key1=value1,key2=value2"); + assertThat(getAttributes()).hasSize(4) + .containsEntry("service.name", "custom-service") + .containsEntry("service.group", "custom-group") + .containsEntry("key1", "value1") + .containsEntry("key2", ""); + } + + @Test + void invalidResourceAttributesShouldBeIgnored() { + this.resourceAttributes.put("", "empty-key"); + this.resourceAttributes.put(null, "null-key"); + this.resourceAttributes.put("null-value", null); + this.resourceAttributes.put("empty-value", ""); + assertThat(getAttributes()).hasSize(2) + .containsEntry("service.name", "unknown_service") + .containsEntry("empty-value", ""); + } + + @Test + @SuppressWarnings("unchecked") + void systemGetEnvShouldBeUsedAsDefaultEnvFunction() { + OpenTelemetryResourceAttributes attributes = new OpenTelemetryResourceAttributes(this.environment, null); + Function getEnv = assertThat(attributes).extracting("getEnv") + .asInstanceOf(InstanceOfAssertFactories.type(Function.class)) + .actual(); + System.getenv().forEach((key, value) -> assertThat(getEnv.apply(key)).isEqualTo(value)); + } + + @Test + void otelResourceAttributeValuesShouldBePercentDecoded() { + PercentEscaper escaper = PercentEscaper.create(); + String value = IntStream.range(32, 127) + .collect(StringBuilder::new, StringBuilder::appendCodePoint, StringBuilder::append) + .toString(); + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "key=" + escaper.escape(value)); + assertThat(getAttributes()).hasSize(2) + .containsEntry("service.name", "unknown_service") + .containsEntry("key", value); + } + + @Test + void otelResourceAttributeValuesShouldBePercentDecodedWhenStringContainsNonAscii() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "key=%20\u015bp\u0159\u00ec\u0144\u0121%20"); + assertThat(getAttributes()).hasSize(2) + .containsEntry("service.name", "unknown_service") + .containsEntry("key", " śpřìńġ "); + } + + @Test + void otelResourceAttributeValuesShouldBePercentDecodedWhenMultiByteSequences() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "key=T%C5%8Dky%C5%8D"); + assertThat(getAttributes()).hasSize(2) + .containsEntry("service.name", "unknown_service") + .containsEntry("key", "Tōkyō"); + } + + @Test + void illegalArgumentExceptionShouldBeThrownWhenDecodingIllegalHexCharPercentEncodedValue() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "key=abc%ß"); + assertThatIllegalArgumentException().isThrownBy(this::getAttributes) + .withMessage("Failed to decode percent-encoded characters at index 3 in the value: 'abc%ß'"); + } + + @Test + void replacementCharShouldBeUsedWhenDecodingNonUtf8Character() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "key=%a3%3e"); + assertThat(getAttributes()).containsEntry("key", "\ufffd>"); + } + + @Test + void illegalArgumentExceptionShouldBeThrownWhenDecodingInvalidPercentEncodedValue() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "key=%"); + assertThatIllegalArgumentException().isThrownBy(this::getAttributes) + .withMessage("Failed to decode percent-encoded characters at index 0 in the value: '%'"); + } + + @Test + void unknownServiceShouldBeUsedAsDefaultServiceName() { + assertThat(getAttributes()).hasSize(1).containsEntry("service.name", "unknown_service"); + } + + @Test + void springApplicationGroupNameShouldBeUsedAsDefaultServiceGroup() { + this.environment.setProperty("spring.application.group", "spring-boot"); + assertThat(getAttributes()).hasSize(3) + .containsEntry("service.name", "unknown_service") + .containsEntry("service.group", "spring-boot") + .containsEntry("service.namespace", "spring-boot"); + } + + @Test + void springApplicationNameShouldBeUsedAsDefaultServiceName() { + this.environment.setProperty("spring.application.name", "spring-boot-app"); + assertThat(getAttributes()).hasSize(1).containsEntry("service.name", "spring-boot-app"); + } + + @Test + void serviceNamespaceShouldNotBePresentByDefault() { + assertThat(getAttributes()).hasSize(1).doesNotContainKey("service.namespace"); + } + + @Test + void resourceAttributesShouldTakePrecedenceOverSpringApplicationName() { + this.resourceAttributes.put("service.name", "spring-boot"); + this.environment.setProperty("spring.application.name", "spring-boot-app"); + assertThat(getAttributes()).hasSize(1).containsEntry("service.name", "spring-boot"); + } + + @Test + void otelResourceAttributesShouldTakePrecedenceOverSpringApplicationName() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "service.name=spring-boot"); + this.environment.setProperty("spring.application.name", "spring-boot-app"); + assertThat(getAttributes()).hasSize(1).containsEntry("service.name", "spring-boot"); + } + + @Test + void otelServiceNameShouldTakePrecedenceOverSpringApplicationName() { + this.environmentVariables.put("OTEL_SERVICE_NAME", "spring-boot"); + this.environment.setProperty("spring.application.name", "spring-boot-app"); + assertThat(getAttributes()).hasSize(1).containsEntry("service.name", "spring-boot"); + } + + @Test + void resourceAttributesShouldTakePrecedenceOverSpringApplicationGroupName() { + this.resourceAttributes.put("service.group", "spring-boot-app"); + this.environment.setProperty("spring.application.group", "spring-boot"); + assertThat(getAttributes()).hasSize(3) + .containsEntry("service.name", "unknown_service") + .containsEntry("service.group", "spring-boot-app"); + } + + @Test + void resourceAttributesShouldTakePrecedenceOverApplicationGroupNameForPopulatingServiceNamespace() { + this.resourceAttributes.put("service.namespace", "spring-boot-app"); + this.environment.setProperty("spring.application.group", "overridden"); + assertThat(getAttributes()).hasSize(3) + .containsEntry("service.name", "unknown_service") + .containsEntry("service.group", "overridden") + .containsEntry("service.namespace", "spring-boot-app"); + } + + @Test + void otelResourceAttributesShouldTakePrecedenceOverSpringApplicationGroupName() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "service.group=spring-boot"); + this.environment.setProperty("spring.application.group", "spring-boot-app"); + assertThat(getAttributes()).hasSize(3) + .containsEntry("service.name", "unknown_service") + .containsEntry("service.group", "spring-boot") + .containsEntry("service.namespace", "spring-boot-app"); + } + + @Test + void otelResourceAttributesShouldTakePrecedenceOverSpringApplicationGroupNameForServiceNamespace() { + this.environmentVariables.put("OTEL_RESOURCE_ATTRIBUTES", "service.namespace=spring-boot"); + this.environment.setProperty("spring.application.group", "overridden"); + assertThat(getAttributes()).hasSize(3) + .containsEntry("service.group", "overridden") + .containsEntry("service.namespace", "spring-boot"); + } + + @Test + void shouldUseServiceGroupForServiceNamespaceIfServiceGroupIsSet() { + this.environment.setProperty("spring.application.group", "alpha"); + assertThat(getAttributes()).containsEntry("service.namespace", "alpha"); + } + + @Test + void shouldNotSetServiceNamespaceIfServiceGroupIsNotSet() { + assertThat(getAttributes()).doesNotContainKey("service.namespace"); + } + + private Map getAttributes() { + Map attributes = new LinkedHashMap<>(); + new OpenTelemetryResourceAttributes(this.environment, this.resourceAttributes, this.environmentVariables::get) + .applyTo(attributes::put); + return attributes; + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/quartz/QuartzEndpointDocumentationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/quartz/QuartzEndpointDocumentationTests.java index ca130aa5dea8..0a93465293a5 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/quartz/QuartzEndpointDocumentationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/quartz/QuartzEndpointDocumentationTests.java @@ -24,6 +24,7 @@ import java.util.Date; import java.util.LinkedHashSet; import java.util.List; +import java.util.Map; import java.util.Map.Entry; import java.util.TimeZone; @@ -55,8 +56,10 @@ import org.springframework.boot.actuate.endpoint.Show; import org.springframework.boot.actuate.quartz.QuartzEndpoint; import org.springframework.boot.actuate.quartz.QuartzEndpointWebExtension; +import org.springframework.boot.json.JsonWriter; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.http.MediaType; import org.springframework.restdocs.payload.FieldDescriptor; import org.springframework.restdocs.payload.JsonFieldType; import org.springframework.scheduling.quartz.DelegatingJob; @@ -68,8 +71,12 @@ import static org.mockito.BDDMockito.given; import static org.mockito.Mockito.mock; import static org.springframework.restdocs.mockmvc.MockMvcRestDocumentation.document; +import static org.springframework.restdocs.operation.preprocess.Preprocessors.preprocessRequest; +import static org.springframework.restdocs.operation.preprocess.Preprocessors.preprocessResponse; +import static org.springframework.restdocs.operation.preprocess.Preprocessors.prettyPrint; import static org.springframework.restdocs.payload.PayloadDocumentation.fieldWithPath; import static org.springframework.restdocs.payload.PayloadDocumentation.relaxedResponseFields; +import static org.springframework.restdocs.payload.PayloadDocumentation.requestFields; import static org.springframework.restdocs.payload.PayloadDocumentation.responseFields; import static org.springframework.restdocs.payload.PayloadDocumentation.subsectionWithPath; @@ -385,6 +392,23 @@ void quartzTriggerCustom() throws Exception { .andWithPrefix("custom.", customTriggerSummary))); } + @Test + void quartzTriggerJob() throws Exception { + mockJobs(jobOne); + String json = JsonWriter.standard().writeToString(Map.of("state", "running")); + assertThat(this.mvc.post() + .content(json) + .contentType(MediaType.APPLICATION_JSON) + .uri("/actuator/quartz/jobs/samples/jobOne")) + .hasStatusOk() + .apply(document("quartz/trigger-job", preprocessRequest(), preprocessResponse(prettyPrint()), + requestFields(fieldWithPath("state").description("The desired state of the job.")), + responseFields(fieldWithPath("group").description("Name of the group."), + fieldWithPath("name").description("Name of the job."), + fieldWithPath("className").description("Fully qualified name of the job implementation."), + fieldWithPath("triggerTime").description("Time the job is triggered.")))); + } + private void setupTriggerDetails(TriggerBuilder builder, TriggerState state) throws SchedulerException { T trigger = builder.withIdentity("example", "samples") diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequestIntegrationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequestIntegrationTests.java new file mode 100644 index 000000000000..d127ab06853e --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequestIntegrationTests.java @@ -0,0 +1,237 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.security.reactive; + +import java.time.Duration; +import java.util.Base64; + +import org.junit.jupiter.api.Test; + +import org.springframework.boot.actuate.autoconfigure.endpoint.EndpointAutoConfiguration; +import org.springframework.boot.actuate.autoconfigure.endpoint.web.WebEndpointAutoConfiguration; +import org.springframework.boot.actuate.autoconfigure.endpoint.web.WebEndpointProperties; +import org.springframework.boot.actuate.autoconfigure.web.server.ManagementContextAutoConfiguration; +import org.springframework.boot.actuate.endpoint.annotation.Endpoint; +import org.springframework.boot.actuate.endpoint.annotation.ReadOperation; +import org.springframework.boot.actuate.endpoint.annotation.WriteOperation; +import org.springframework.boot.autoconfigure.AutoConfigurations; +import org.springframework.boot.autoconfigure.http.HttpMessageConvertersAutoConfiguration; +import org.springframework.boot.autoconfigure.jackson.JacksonAutoConfiguration; +import org.springframework.boot.autoconfigure.security.reactive.ReactiveSecurityAutoConfiguration; +import org.springframework.boot.autoconfigure.security.reactive.ReactiveUserDetailsServiceAutoConfiguration; +import org.springframework.boot.autoconfigure.web.reactive.HttpHandlerAutoConfiguration; +import org.springframework.boot.autoconfigure.web.reactive.WebFluxAutoConfiguration; +import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.boot.test.context.assertj.AssertableReactiveWebApplicationContext; +import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner; +import org.springframework.boot.web.embedded.tomcat.TomcatReactiveWebServerFactory; +import org.springframework.boot.web.reactive.context.AnnotationConfigReactiveWebServerApplicationContext; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.http.HttpMethod; +import org.springframework.security.config.Customizer; +import org.springframework.security.config.web.server.ServerHttpSecurity; +import org.springframework.security.config.web.server.ServerHttpSecurity.CsrfSpec; +import org.springframework.security.core.userdetails.MapReactiveUserDetailsService; +import org.springframework.security.core.userdetails.User; +import org.springframework.security.web.server.SecurityWebFilterChain; +import org.springframework.test.web.reactive.server.WebTestClient; + +/** + * Integration tests for {@link EndpointRequest}. + * + * @author Chris Bono + */ +class EndpointRequestIntegrationTests { + + @Test + void toEndpointShouldMatch() { + getContextRunner().run((context) -> { + WebTestClient webTestClient = getWebTestClient(context); + webTestClient.get().uri("/actuator/e1").exchange().expectStatus().isOk(); + }); + } + + @Test + void toEndpointPostShouldMatch() { + getContextRunner().withPropertyValues("spring.security.user.password=password").run((context) -> { + WebTestClient webTestClient = getWebTestClient(context); + webTestClient.post().uri("/actuator/e1").exchange().expectStatus().isUnauthorized(); + webTestClient.post() + .uri("/actuator/e1") + .header("Authorization", getBasicAuth()) + .exchange() + .expectStatus() + .isNoContent(); + }); + } + + @Test + void toAllEndpointsShouldMatch() { + getContextRunner().withPropertyValues("spring.security.user.password=password").run((context) -> { + WebTestClient webTestClient = getWebTestClient(context); + webTestClient.get().uri("/actuator/e2").exchange().expectStatus().isUnauthorized(); + webTestClient.get() + .uri("/actuator/e2") + .header("Authorization", getBasicAuth()) + .exchange() + .expectStatus() + .isOk(); + }); + } + + @Test + void toLinksShouldMatch() { + getContextRunner().run((context) -> { + WebTestClient webTestClient = getWebTestClient(context); + webTestClient.get().uri("/actuator").exchange().expectStatus().isOk(); + }); + } + + protected final ReactiveWebApplicationContextRunner getContextRunner() { + return createContextRunner().withPropertyValues("management.endpoints.web.exposure.include=*") + .withUserConfiguration(BaseConfiguration.class, SecurityConfiguration.class) + .withConfiguration( + AutoConfigurations.of(JacksonAutoConfiguration.class, ReactiveSecurityAutoConfiguration.class, + ReactiveUserDetailsServiceAutoConfiguration.class, EndpointAutoConfiguration.class, + WebEndpointAutoConfiguration.class, ManagementContextAutoConfiguration.class)); + + } + + protected ReactiveWebApplicationContextRunner createContextRunner() { + return new ReactiveWebApplicationContextRunner(AnnotationConfigReactiveWebServerApplicationContext::new) + .withUserConfiguration(WebEndpointConfiguration.class) + .withConfiguration(AutoConfigurations.of(HttpHandlerAutoConfiguration.class, + HttpMessageConvertersAutoConfiguration.class, WebFluxAutoConfiguration.class)); + } + + protected WebTestClient getWebTestClient(AssertableReactiveWebApplicationContext context) { + int port = context.getSourceApplicationContext(AnnotationConfigReactiveWebServerApplicationContext.class) + .getWebServer() + .getPort(); + return WebTestClient.bindToServer() + .baseUrl("http://localhost:" + port) + .responseTimeout(Duration.ofMinutes(5)) + .build(); + } + + private String getBasicAuth() { + return "Basic " + Base64.getEncoder().encodeToString("user:password".getBytes()); + } + + @Configuration(proxyBeanMethods = false) + static class BaseConfiguration { + + @Bean + TestEndpoint1 endpoint1() { + return new TestEndpoint1(); + } + + @Bean + TestEndpoint2 endpoint2() { + return new TestEndpoint2(); + } + + @Bean + TestEndpoint3 endpoint3() { + return new TestEndpoint3(); + } + + } + + @Endpoint(id = "e1") + static class TestEndpoint1 { + + @ReadOperation + Object getAll() { + return "endpoint 1"; + } + + @WriteOperation + void setAll() { + } + + } + + @Endpoint(id = "e2") + static class TestEndpoint2 { + + @ReadOperation + Object getAll() { + return "endpoint 2"; + } + + } + + @Endpoint(id = "e3") + static class TestEndpoint3 { + + @ReadOperation + Object getAll() { + return null; + } + + } + + @Configuration(proxyBeanMethods = false) + @EnableConfigurationProperties(WebEndpointProperties.class) + static class WebEndpointConfiguration { + + @Bean + TomcatReactiveWebServerFactory tomcat() { + return new TomcatReactiveWebServerFactory(0); + } + + } + + @Configuration(proxyBeanMethods = false) + static class SecurityConfiguration { + + @SuppressWarnings("deprecation") + @Bean + MapReactiveUserDetailsService userDetailsService() { + return new MapReactiveUserDetailsService( + User.withDefaultPasswordEncoder() + .username("user") + .password("password") + .authorities("ROLE_USER") + .build(), + User.withDefaultPasswordEncoder() + .username("admin") + .password("admin") + .authorities("ROLE_ACTUATOR", "ROLE_USER") + .build()); + } + + @Bean + SecurityWebFilterChain springSecurityFilterChain(ServerHttpSecurity http) { + http.authorizeExchange((exchanges) -> { + exchanges.matchers(EndpointRequest.toLinks()).permitAll(); + exchanges.matchers(EndpointRequest.to(TestEndpoint1.class).withHttpMethod(HttpMethod.POST)) + .authenticated(); + exchanges.matchers(EndpointRequest.to(TestEndpoint1.class)).permitAll(); + exchanges.matchers(EndpointRequest.toAnyEndpoint()).authenticated(); + exchanges.anyExchange().hasRole("ADMIN"); + }); + http.httpBasic(Customizer.withDefaults()); + http.csrf(CsrfSpec::disable); + return http.build(); + } + + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequestTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequestTests.java index 02e5402a6567..18ac34841d0b 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequestTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/reactive/EndpointRequestTests.java @@ -35,6 +35,7 @@ import org.springframework.boot.web.context.WebServerApplicationContext; import org.springframework.boot.web.server.WebServer; import org.springframework.context.support.StaticApplicationContext; +import org.springframework.http.HttpMethod; import org.springframework.http.server.reactive.ServerHttpRequest; import org.springframework.http.server.reactive.ServerHttpResponse; import org.springframework.mock.http.server.reactive.MockServerHttpRequest; @@ -54,6 +55,7 @@ * * @author Madhura Bhave * @author Phillip Webb + * @author Chris Bono */ class EndpointRequestTests { @@ -65,6 +67,13 @@ void toAnyEndpointShouldMatchEndpointPath() { assertMatcher(matcher).matches("/actuator"); } + @Test + void toAnyEndpointWithHttpMethodShouldRespectRequestMethod() { + ServerWebExchangeMatcher matcher = EndpointRequest.toAnyEndpoint().withHttpMethod(HttpMethod.POST); + assertMatcher(matcher, "/actuator").matches(HttpMethod.POST, "/actuator/foo"); + assertMatcher(matcher, "/actuator").doesNotMatch(HttpMethod.GET, "/actuator/foo"); + } + @Test void toAnyEndpointShouldMatchEndpointPathWithTrailingSlash() { ServerWebExchangeMatcher matcher = EndpointRequest.toAnyEndpoint(); @@ -390,6 +399,12 @@ void matches(String path) { matches(exchange); } + void matches(HttpMethod httpMethod, String path) { + ServerWebExchange exchange = webHandler() + .createExchange(MockServerHttpRequest.method(httpMethod, path).build(), new MockServerHttpResponse()); + matches(exchange); + } + private void matches(ServerWebExchange exchange) { assertThat(this.matcher.matches(exchange).block(Duration.ofSeconds(30)).isMatch()) .as("Matches " + getRequestPath(exchange)) @@ -402,6 +417,12 @@ void doesNotMatch(String path) { doesNotMatch(exchange); } + void doesNotMatch(HttpMethod httpMethod, String path) { + ServerWebExchange exchange = webHandler() + .createExchange(MockServerHttpRequest.method(httpMethod, path).build(), new MockServerHttpResponse()); + doesNotMatch(exchange); + } + private void doesNotMatch(ServerWebExchange exchange) { assertThat(this.matcher.matches(exchange).block(Duration.ofSeconds(30)).isMatch()) .as("Does not match " + getRequestPath(exchange)) diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/AbstractEndpointRequestIntegrationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/AbstractEndpointRequestIntegrationTests.java index 6a2ca3dbb028..ea0ade75d2a6 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/AbstractEndpointRequestIntegrationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/AbstractEndpointRequestIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,6 +32,7 @@ import org.springframework.boot.actuate.autoconfigure.web.server.ManagementContextAutoConfiguration; import org.springframework.boot.actuate.endpoint.annotation.Endpoint; import org.springframework.boot.actuate.endpoint.annotation.ReadOperation; +import org.springframework.boot.actuate.endpoint.annotation.WriteOperation; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.jackson.JacksonAutoConfiguration; import org.springframework.boot.autoconfigure.security.servlet.SecurityAutoConfiguration; @@ -40,8 +41,10 @@ import org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.http.HttpMethod; import org.springframework.http.HttpStatus; import org.springframework.security.config.annotation.web.builders.HttpSecurity; +import org.springframework.security.config.annotation.web.configurers.CsrfConfigurer; import org.springframework.security.core.userdetails.User; import org.springframework.security.provisioning.InMemoryUserDetailsManager; import org.springframework.security.web.SecurityFilterChain; @@ -53,6 +56,7 @@ * Abstract base class for {@link EndpointRequest} tests. * * @author Madhura Bhave + * @author Chris Bono */ abstract class AbstractEndpointRequestIntegrationTests { @@ -64,6 +68,20 @@ void toEndpointShouldMatch() { }); } + @Test + void toEndpointPostShouldMatch() { + getContextRunner().withPropertyValues("spring.security.user.password=password").run((context) -> { + WebTestClient webTestClient = getWebTestClient(context); + webTestClient.post().uri("/actuator/e1").exchange().expectStatus().isUnauthorized(); + webTestClient.post() + .uri("/actuator/e1") + .header("Authorization", getBasicAuth()) + .exchange() + .expectStatus() + .isNoContent(); + }); + } + @Test void toAllEndpointsShouldMatch() { getContextRunner().withPropertyValues("spring.security.user.password=password").run((context) -> { @@ -153,6 +171,10 @@ Object getAll() { return "endpoint 1"; } + @WriteOperation + void setAll() { + } + } @Endpoint(id = "e2") @@ -200,10 +222,13 @@ InMemoryUserDetailsManager userDetailsManager() { SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception { http.authorizeHttpRequests((requests) -> { requests.requestMatchers(EndpointRequest.toLinks()).permitAll(); + requests.requestMatchers(EndpointRequest.to(TestEndpoint1.class).withHttpMethod(HttpMethod.POST)) + .authenticated(); requests.requestMatchers(EndpointRequest.to(TestEndpoint1.class)).permitAll(); requests.requestMatchers(EndpointRequest.toAnyEndpoint()).authenticated(); requests.anyRequest().hasRole("ADMIN"); }); + http.csrf(CsrfConfigurer::disable); http.httpBasic(withDefaults()); return http.build(); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/EndpointRequestTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/EndpointRequestTests.java index 101cec3d91a3..3c112666effe 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/EndpointRequestTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/EndpointRequestTests.java @@ -34,9 +34,9 @@ import org.springframework.boot.actuate.endpoint.web.PathMappedEndpoint; import org.springframework.boot.actuate.endpoint.web.PathMappedEndpoints; import org.springframework.boot.actuate.endpoint.web.WebServerNamespace; -import org.springframework.boot.autoconfigure.security.servlet.RequestMatcherProvider; import org.springframework.boot.web.context.WebServerApplicationContext; import org.springframework.boot.web.server.WebServer; +import org.springframework.http.HttpMethod; import org.springframework.mock.web.MockHttpServletRequest; import org.springframework.mock.web.MockServletContext; import org.springframework.security.web.util.matcher.RequestMatcher; @@ -52,6 +52,7 @@ * * @author Phillip Webb * @author Madhura Bhave + * @author Chris Bono */ class EndpointRequestTests { @@ -65,6 +66,14 @@ void toAnyEndpointShouldMatchEndpointPath() { assertMatcher(matcher, "/actuator").matches("/actuator"); } + @Test + void toAnyEndpointWithHttpMethodShouldRespectRequestMethod() { + EndpointRequest.EndpointRequestMatcher matcher = EndpointRequest.toAnyEndpoint() + .withHttpMethod(HttpMethod.POST); + assertMatcher(matcher, "/actuator").matches(HttpMethod.POST, "/actuator/foo"); + assertMatcher(matcher, "/actuator").doesNotMatch(HttpMethod.GET, "/actuator/foo"); + } + @Test void toAnyEndpointShouldMatchEndpointPathWithTrailingSlash() { RequestMatcher matcher = EndpointRequest.toAnyEndpoint(); @@ -199,7 +208,7 @@ void endpointRequestMatcherShouldUseCustomRequestMatcherProvider() { RequestMatcher matcher = EndpointRequest.toAnyEndpoint(); RequestMatcher mockRequestMatcher = (request) -> false; RequestMatcherAssert assertMatcher = assertMatcher(matcher, mockPathMappedEndpoints(""), - (pattern) -> mockRequestMatcher, null); + (pattern, method) -> mockRequestMatcher, null); assertMatcher.doesNotMatch("/foo"); assertMatcher.doesNotMatch("/bar"); } @@ -209,7 +218,7 @@ void linksRequestMatcherShouldUseCustomRequestMatcherProvider() { RequestMatcher matcher = EndpointRequest.toLinks(); RequestMatcher mockRequestMatcher = (request) -> false; RequestMatcherAssert assertMatcher = assertMatcher(matcher, mockPathMappedEndpoints("/actuator"), - (pattern) -> mockRequestMatcher, null); + (pattern, method) -> mockRequestMatcher, null); assertMatcher.doesNotMatch("/actuator"); } @@ -393,7 +402,11 @@ static class RequestMatcherAssert implements AssertDelegateTarget { } void matches(String servletPath) { - matches(mockRequest(servletPath)); + matches(mockRequest(null, servletPath)); + } + + void matches(HttpMethod httpMethod, String servletPath) { + matches(mockRequest(httpMethod, servletPath)); } private void matches(HttpServletRequest request) { @@ -401,20 +414,27 @@ private void matches(HttpServletRequest request) { } void doesNotMatch(String servletPath) { - doesNotMatch(mockRequest(servletPath)); + doesNotMatch(mockRequest(null, servletPath)); + } + + void doesNotMatch(HttpMethod httpMethod, String servletPath) { + doesNotMatch(mockRequest(httpMethod, servletPath)); } private void doesNotMatch(HttpServletRequest request) { assertThat(this.matcher.matches(request)).as("Does not match " + getRequestPath(request)).isFalse(); } - private MockHttpServletRequest mockRequest(String servletPath) { + private MockHttpServletRequest mockRequest(HttpMethod httpMethod, String servletPath) { MockServletContext servletContext = new MockServletContext(); servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context); MockHttpServletRequest request = new MockHttpServletRequest(servletContext); if (servletPath != null) { request.setServletPath(servletPath); } + if (httpMethod != null) { + request.setMethod(httpMethod.name()); + } return request; } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/SecurityRequestMatchersManagementContextConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/SecurityRequestMatchersManagementContextConfigurationTests.java index c10a3c24a71e..2423fff04291 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/SecurityRequestMatchersManagementContextConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/security/servlet/SecurityRequestMatchersManagementContextConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,8 +19,6 @@ import org.junit.jupiter.api.Test; import org.springframework.boot.autoconfigure.AutoConfigurations; -import org.springframework.boot.autoconfigure.security.servlet.AntPathRequestMatcherProvider; -import org.springframework.boot.autoconfigure.security.servlet.RequestMatcherProvider; import org.springframework.boot.autoconfigure.web.servlet.DispatcherServletPath; import org.springframework.boot.autoconfigure.web.servlet.JerseyApplicationPath; import org.springframework.boot.test.context.FilteredClassLoader; @@ -61,7 +59,7 @@ void configurationConditionalOnRequestMatcherClass() { void registersRequestMatcherProviderIfMvcPresent() { this.contextRunner.withUserConfiguration(TestMvcConfiguration.class).run((context) -> { AntPathRequestMatcherProvider matcherProvider = context.getBean(AntPathRequestMatcherProvider.class); - RequestMatcher requestMatcher = matcherProvider.getRequestMatcher("/example"); + RequestMatcher requestMatcher = matcherProvider.getRequestMatcher("/example", null); assertThat(requestMatcher).extracting("pattern").isEqualTo("/custom/example"); }); } @@ -72,7 +70,7 @@ void registersRequestMatcherForJerseyProviderIfJerseyPresentAndMvcAbsent() { .withUserConfiguration(TestJerseyConfiguration.class) .run((context) -> { AntPathRequestMatcherProvider matcherProvider = context.getBean(AntPathRequestMatcherProvider.class); - RequestMatcher requestMatcher = matcherProvider.getRequestMatcher("/example"); + RequestMatcher requestMatcher = matcherProvider.getRequestMatcher("/example", null); assertThat(requestMatcher).extracting("pattern").isEqualTo("/admin/example"); }); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/ssl/SslMeterBinderTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/ssl/SslMeterBinderTests.java new file mode 100644 index 000000000000..0d3d0057bcfa --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/ssl/SslMeterBinderTests.java @@ -0,0 +1,107 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.ssl; + +import java.time.Clock; +import java.time.Duration; +import java.time.Instant; +import java.time.ZoneId; + +import io.micrometer.core.instrument.MeterRegistry; +import io.micrometer.core.instrument.simple.SimpleMeterRegistry; +import org.junit.jupiter.api.Test; + +import org.springframework.boot.info.SslInfo; +import org.springframework.boot.ssl.DefaultSslBundleRegistry; +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslBundles; +import org.springframework.boot.ssl.SslStoreBundle; +import org.springframework.boot.ssl.jks.JksSslStoreBundle; +import org.springframework.boot.ssl.jks.JksSslStoreDetails; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link SslMeterBinder}. + * + * @author Moritz Halbritter + */ +class SslMeterBinderTests { + + private static final Clock CLOCK = Clock.fixed(Instant.parse("2024-10-21T13:51:40Z"), ZoneId.of("UTC")); + + @Test + void shouldRegisterChainMetrics() { + MeterRegistry meterRegistry = bindToRegistry(); + assertThat(meterRegistry.get("ssl.chains").tag("status", "valid").gauge().value()).isEqualTo(3.0); + assertThat(meterRegistry.get("ssl.chains").tag("status", "expired").gauge().value()).isEqualTo(1.0); + assertThat(meterRegistry.get("ssl.chains").tag("status", "not-yet-valid").gauge().value()).isEqualTo(1.0); + assertThat(meterRegistry.get("ssl.chains").tag("status", "will-expire-soon").gauge().value()).isEqualTo(0.0); + } + + @Test + void shouldRegisterChainExpiryMetrics() { + MeterRegistry meterRegistry = bindToRegistry(); + assertThat(Duration.ofSeconds(findExpiryGauge(meterRegistry, "ca", "419224ce190242b2c44069dd3c560192b3b669f3"))) + .hasDays(1095); + assertThat(Duration + .ofSeconds(findExpiryGauge(meterRegistry, "intermediary", "60f79365fc46bf69149754d377680192b3b6bcf5"))) + .hasDays(730); + assertThat(Duration + .ofSeconds(findExpiryGauge(meterRegistry, "server", "504c45129526ac050abb11459b1f0192b3b70fe9"))) + .hasDays(365); + assertThat(Duration + .ofSeconds(findExpiryGauge(meterRegistry, "expired", "562bc5dcf4f26bb179abb13068180192b3bb53dc"))) + .hasDays(-386); + assertThat(Duration + .ofSeconds(findExpiryGauge(meterRegistry, "not-yet-valid", "7df79335f274e2cfa7467fd5f9ce0192b3bcf4aa"))) + .hasDays(36889); + } + + private static long findExpiryGauge(MeterRegistry meterRegistry, String chain, String certificateSerialNumber) { + return (long) meterRegistry.get("ssl.chain.expiry") + .tag("bundle", "test-0") + .tag("chain", chain) + .tag("certificate", certificateSerialNumber) + .gauge() + .value(); + } + + private SimpleMeterRegistry bindToRegistry() { + SslBundles sslBundles = createSslBundles("classpath:certificates/chains.p12"); + SslInfo sslInfo = createSslInfo(sslBundles); + SslMeterBinder binder = new SslMeterBinder(sslInfo, sslBundles, CLOCK); + SimpleMeterRegistry meterRegistry = new SimpleMeterRegistry(); + binder.bindTo(meterRegistry); + return meterRegistry; + } + + private SslBundles createSslBundles(String... locations) { + DefaultSslBundleRegistry sslBundleRegistry = new DefaultSslBundleRegistry(); + for (int i = 0; i < locations.length; i++) { + JksSslStoreDetails keyStoreDetails = JksSslStoreDetails.forLocation(locations[i]).withPassword("secret"); + SslStoreBundle sslStoreBundle = new JksSslStoreBundle(keyStoreDetails, null); + sslBundleRegistry.registerBundle("test-%d".formatted(i), SslBundle.of(sslStoreBundle)); + } + return sslBundleRegistry; + } + + private SslInfo createSslInfo(SslBundles sslBundles) { + return new SslInfo(sslBundles, Duration.ofDays(7), CLOCK); + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/ssl/SslObservabilityAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/ssl/SslObservabilityAutoConfigurationTests.java new file mode 100644 index 000000000000..80face585611 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/ssl/SslObservabilityAutoConfigurationTests.java @@ -0,0 +1,66 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.ssl; + +import org.junit.jupiter.api.Test; + +import org.springframework.boot.actuate.autoconfigure.metrics.CompositeMeterRegistryAutoConfiguration; +import org.springframework.boot.actuate.autoconfigure.metrics.MetricsAutoConfiguration; +import org.springframework.boot.autoconfigure.AutoConfigurations; +import org.springframework.boot.autoconfigure.ssl.SslAutoConfiguration; +import org.springframework.boot.info.SslInfo; +import org.springframework.boot.test.context.runner.ApplicationContextRunner; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link SslObservabilityAutoConfiguration}. + * + * @author Moritz Halbritter + */ +class SslObservabilityAutoConfigurationTests { + + private final ApplicationContextRunner contextRunner = new ApplicationContextRunner().withConfiguration( + AutoConfigurations.of(MetricsAutoConfiguration.class, CompositeMeterRegistryAutoConfiguration.class, + SslAutoConfiguration.class, SslObservabilityAutoConfiguration.class)); + + private final ApplicationContextRunner contextRunnerWithoutSslBundles = new ApplicationContextRunner() + .withConfiguration(AutoConfigurations.of(MetricsAutoConfiguration.class, + CompositeMeterRegistryAutoConfiguration.class, SslObservabilityAutoConfiguration.class)); + + private final ApplicationContextRunner contextRunnerWithoutMeterRegistry = new ApplicationContextRunner() + .withConfiguration(AutoConfigurations.of(SslAutoConfiguration.class, SslObservabilityAutoConfiguration.class)); + + @Test + void shouldSupplyBeans() { + this.contextRunner + .run((context) -> assertThat(context).hasSingleBean(SslMeterBinder.class).hasSingleBean(SslInfo.class)); + } + + @Test + void shouldBackOffIfSslBundlesIsMissing() { + this.contextRunnerWithoutSslBundles + .run((context) -> assertThat(context).doesNotHaveBean(SslMeterBinder.class).doesNotHaveBean(SslInfo.class)); + } + + @Test + void shouldBackOffIfMeterRegistryIsMissing() { + this.contextRunnerWithoutMeterRegistry + .run((context) -> assertThat(context).doesNotHaveBean(SslMeterBinder.class).doesNotHaveBean(SslInfo.class)); + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/system/DiskSpaceHealthContributorAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/system/DiskSpaceHealthContributorAutoConfigurationTests.java index 20b28b81b61c..18675cfbb124 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/system/DiskSpaceHealthContributorAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/system/DiskSpaceHealthContributorAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,7 @@ void thresholdMustBePositive() { .run((context) -> assertThat(context).hasFailed() .getFailure() .rootCause() - .hasMessage("threshold must be greater than or equal to 0")); + .hasMessage("'threshold' must be greater than or equal to 0")); } @Test diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/BaggagePropagationIntegrationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/BaggagePropagationIntegrationTests.java index ae70c4cb9efd..0175feb9a2ef 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/BaggagePropagationIntegrationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/BaggagePropagationIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/MicrometerTracingAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/MicrometerTracingAutoConfigurationTests.java index 00a3a3977bbc..8f6cbffee3a3 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/MicrometerTracingAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/MicrometerTracingAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -145,18 +145,6 @@ void shouldNotSupplyAspectBeansIfPropertyIsDisabled() { }); } - @Test - void shouldSupplyAspectBeansIfLegacyPropertyIsEnabled() { - new ApplicationContextRunner().withPropertyValues("micrometer.observations.annotations.enabled=true") - .withConfiguration(AutoConfigurations.of(MicrometerTracingAutoConfiguration.class)) - .withUserConfiguration(TracerConfiguration.class, PropagatorConfiguration.class) - .run((context) -> { - assertThat(context).hasSingleBean(DefaultNewSpanParser.class); - assertThat(context).hasSingleBean(ImperativeMethodInvocationProcessor.class); - assertThat(context).hasSingleBean(SpanAspect.class); - }); - } - @Test void shouldNotSupplyBeansIfAspectjIsMissing() { this.contextRunner.withUserConfiguration(TracerConfiguration.class) diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryTracingAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryTracingAutoConfigurationTests.java index c38e2fc11313..38dd0c009193 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryTracingAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/OpenTelemetryTracingAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -51,6 +51,7 @@ import io.opentelemetry.sdk.trace.SpanLimits; import io.opentelemetry.sdk.trace.SpanProcessor; import io.opentelemetry.sdk.trace.data.SpanData; +import io.opentelemetry.sdk.trace.export.BatchSpanProcessor; import io.opentelemetry.sdk.trace.export.SpanExporter; import io.opentelemetry.sdk.trace.samplers.Sampler; import org.assertj.core.api.InstanceOfAssertFactories; @@ -69,6 +70,7 @@ import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.core.annotation.Order; +import org.springframework.test.util.ReflectionTestUtils; import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.fail; @@ -172,6 +174,8 @@ void shouldBackOffOnCustomBeans() { assertThat(context).hasSingleBean(SpanProcessors.class); assertThat(context).hasBean("customSpanExporters"); assertThat(context).hasSingleBean(SpanExporters.class); + assertThat(context).hasBean("customBatchSpanProcessor"); + assertThat(context).hasSingleBean(BatchSpanProcessor.class); }); } @@ -321,6 +325,44 @@ void shouldDisablePropagationIfTracingIsDisabled() { }); } + @Test + void batchSpanProcessorShouldBeConfiguredWithCustomProperties() { + this.contextRunner + .withPropertyValues("management.tracing.opentelemetry.export.timeout=45s", + "management.tracing.opentelemetry.export.include-unsampled=true", + "management.tracing.opentelemetry.export.max-batch-size=256", + "management.tracing.opentelemetry.export.max-queue-size=4096", + "management.tracing.opentelemetry.export.schedule-delay=15s") + .run((context) -> { + assertThat(context).hasSingleBean(BatchSpanProcessor.class); + BatchSpanProcessor batchSpanProcessor = context.getBean(BatchSpanProcessor.class); + assertThat(batchSpanProcessor).hasFieldOrPropertyWithValue("exportUnsampledSpans", true) + .extracting("worker") + .hasFieldOrPropertyWithValue("exporterTimeoutNanos", Duration.ofSeconds(45).toNanos()) + .hasFieldOrPropertyWithValue("maxExportBatchSize", 256) + .hasFieldOrPropertyWithValue("scheduleDelayNanos", Duration.ofSeconds(15).toNanos()) + .extracting("queue") + .satisfies((queue) -> assertThat(ReflectionTestUtils.invokeMethod(queue, "capacity")) + .isEqualTo(4096)); + }); + } + + @Test + void batchSpanProcessorShouldBeConfiguredWithDefaultProperties() { + this.contextRunner.run((context) -> { + assertThat(context).hasSingleBean(BatchSpanProcessor.class); + BatchSpanProcessor batchSpanProcessor = context.getBean(BatchSpanProcessor.class); + assertThat(batchSpanProcessor).hasFieldOrPropertyWithValue("exportUnsampledSpans", false) + .extracting("worker") + .hasFieldOrPropertyWithValue("exporterTimeoutNanos", Duration.ofSeconds(30).toNanos()) + .hasFieldOrPropertyWithValue("maxExportBatchSize", 512) + .hasFieldOrPropertyWithValue("scheduleDelayNanos", Duration.ofSeconds(5).toNanos()) + .extracting("queue") + .satisfies((queue) -> assertThat(ReflectionTestUtils.invokeMethod(queue, "capacity")) + .isEqualTo(2048)); + }); + } + @Test // gh-41439 @ForkedClassPath void shouldPublishEventsWhenContextStorageIsInitializedEarly() { @@ -401,6 +443,11 @@ SpanExporter spanExporter2() { @Configuration(proxyBeanMethods = false) private static final class CustomConfiguration { + @Bean + BatchSpanProcessor customBatchSpanProcessor() { + return mock(BatchSpanProcessor.class); + } + @Bean SpanProcessors customSpanProcessors() { return SpanProcessors.of(mock(SpanProcessor.class)); diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingAutoConfigurationTests.java index 174592638a7d..00a4ad8b15c5 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/otlp/OtlpTracingAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,10 +16,19 @@ package org.springframework.boot.actuate.autoconfigure.tracing.otlp; +import java.time.Duration; +import java.util.List; +import java.util.function.Supplier; + +import io.opentelemetry.api.metrics.MeterProvider; +import io.opentelemetry.exporter.internal.compression.GzipCompressor; import io.opentelemetry.exporter.otlp.http.trace.OtlpHttpSpanExporter; +import io.opentelemetry.exporter.otlp.http.trace.OtlpHttpSpanExporterBuilder; import io.opentelemetry.exporter.otlp.trace.OtlpGrpcSpanExporter; +import io.opentelemetry.exporter.otlp.trace.OtlpGrpcSpanExporterBuilder; import io.opentelemetry.sdk.trace.export.SpanExporter; import okhttp3.HttpUrl; +import org.assertj.core.api.InstanceOfAssertFactories; import org.junit.jupiter.api.Test; import org.springframework.boot.actuate.autoconfigure.tracing.otlp.OtlpTracingConfigurations.ConnectionDetails.PropertiesOtlpTracingConnectionDetails; @@ -28,6 +37,7 @@ import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.test.util.ReflectionTestUtils; import static org.assertj.core.api.Assertions.assertThat; @@ -64,6 +74,27 @@ void shouldSupplyBeans() { .hasSingleBean(SpanExporter.class)); } + @Test + void shouldCustomizeHttpTransportWithProperties() { + this.contextRunner + .withPropertyValues("management.otlp.tracing.endpoint=http://localhost:4317/v1/traces", + "management.otlp.tracing.timeout=10m", "management.otlp.tracing.connect-timeout=20m", + "management.otlp.tracing.compression=GZIP", "management.otlp.tracing.headers.spring=boot") + .run((context) -> { + assertThat(context).hasSingleBean(OtlpHttpSpanExporter.class).hasSingleBean(SpanExporter.class); + OtlpHttpSpanExporter exporter = context.getBean(OtlpHttpSpanExporter.class); + assertThat(exporter).extracting("delegate.httpSender.client") + .hasFieldOrPropertyWithValue("connectTimeoutMillis", 1200000) + .hasFieldOrPropertyWithValue("callTimeoutMillis", 600000); + assertThat(exporter).extracting("delegate.httpSender.compressor").isInstanceOf(GzipCompressor.class); + assertThat(exporter).extracting("delegate.httpSender.headerSupplier") + .asInstanceOf(InstanceOfAssertFactories.type(Supplier.class)) + .satisfies((headerSupplier) -> assertThat(headerSupplier.get()) + .asInstanceOf(InstanceOfAssertFactories.map(String.class, List.class)) + .containsEntry("spring", List.of("boot"))); + }); + } + @Test void shouldSupplyBeansIfGrpcTransportIsEnabled() { this.contextRunner @@ -73,6 +104,28 @@ void shouldSupplyBeansIfGrpcTransportIsEnabled() { .hasSingleBean(SpanExporter.class)); } + @Test + void shouldCustomizeGrpcTransportWithProperties() { + this.contextRunner + .withPropertyValues("management.otlp.tracing.endpoint=http://localhost:4317/v1/traces", + "management.otlp.tracing.transport=grpc", "management.otlp.tracing.timeout=10m", + "management.otlp.tracing.connect-timeout=20m", "management.otlp.tracing.compression=GZIP", + "management.otlp.tracing.headers.spring=boot") + .run((context) -> { + assertThat(context).hasSingleBean(OtlpGrpcSpanExporter.class).hasSingleBean(SpanExporter.class); + OtlpGrpcSpanExporter exporter = context.getBean(OtlpGrpcSpanExporter.class); + assertThat(exporter).extracting("delegate.grpcSender.client") + .hasFieldOrPropertyWithValue("connectTimeoutMillis", 1200000) + .hasFieldOrPropertyWithValue("callTimeoutMillis", 600000); + assertThat(exporter).extracting("delegate.grpcSender.compressor").isInstanceOf(GzipCompressor.class); + assertThat(exporter).extracting("delegate.grpcSender.headersSupplier") + .asInstanceOf(InstanceOfAssertFactories.type(Supplier.class)) + .satisfies((headerSupplier) -> assertThat(headerSupplier.get()) + .asInstanceOf(InstanceOfAssertFactories.map(String.class, List.class)) + .containsEntry("spring", List.of("boot"))); + }); + } + @Test void shouldNotSupplyBeansIfGlobalTracingIsDisabled() { this.contextRunner.withPropertyValues("management.tracing.enabled=false") @@ -147,6 +200,88 @@ void testConnectionFactoryWithOverridesWhenUsingCustomConnectionDetails() { }); } + @Test + @SuppressWarnings("unchecked") + void httpShouldUseMeterProviderIfSet() { + this.contextRunner.withUserConfiguration(MeterProviderConfiguration.class) + .withPropertyValues("management.otlp.tracing.endpoint=http://localhost:4318/v1/traces") + .run((context) -> { + OtlpHttpSpanExporter otlpHttpSpanExporter = context.getBean(OtlpHttpSpanExporter.class); + OtlpHttpSpanExporterBuilder builder = otlpHttpSpanExporter.toBuilder(); + Supplier meterProviderSupplier = (Supplier) ReflectionTestUtils + .getField(ReflectionTestUtils.getField(builder, "delegate"), "meterProviderSupplier"); + assertThat(meterProviderSupplier).isNotNull(); + assertThat(meterProviderSupplier.get()).isSameAs(MeterProviderConfiguration.meterProvider); + }); + } + + @Test + @SuppressWarnings("unchecked") + void grpcShouldUseMeterProviderIfSet() { + this.contextRunner.withUserConfiguration(MeterProviderConfiguration.class) + .withPropertyValues("management.otlp.tracing.endpoint=http://localhost:4318/v1/traces", + "management.otlp.tracing.transport=grpc") + .run((context) -> { + OtlpGrpcSpanExporter otlpGrpcSpanExporter = context.getBean(OtlpGrpcSpanExporter.class); + OtlpGrpcSpanExporterBuilder builder = otlpGrpcSpanExporter.toBuilder(); + Supplier meterProviderSupplier = (Supplier) ReflectionTestUtils + .getField(ReflectionTestUtils.getField(builder, "delegate"), "meterProviderSupplier"); + assertThat(meterProviderSupplier).isNotNull(); + assertThat(meterProviderSupplier.get()).isSameAs(MeterProviderConfiguration.meterProvider); + }); + } + + @Test + void shouldCustomizeHttpTransportWithOtlpHttpSpanExporterBuilderCustomizer() { + Duration connectTimeout = Duration.ofMinutes(20); + Duration timeout = Duration.ofMinutes(10); + this.contextRunner + .withBean("httpCustomizer1", OtlpHttpSpanExporterBuilderCustomizer.class, + () -> (builder) -> builder.setConnectTimeout(connectTimeout)) + .withBean("httpCustomizer2", OtlpHttpSpanExporterBuilderCustomizer.class, + () -> (builder) -> builder.setTimeout(timeout)) + .withPropertyValues("management.otlp.tracing.endpoint=http://localhost:4317/v1/traces") + .run((context) -> { + assertThat(context).hasSingleBean(OtlpHttpSpanExporter.class).hasSingleBean(SpanExporter.class); + OtlpHttpSpanExporter exporter = context.getBean(OtlpHttpSpanExporter.class); + assertThat(exporter).extracting("delegate.httpSender.client") + .hasFieldOrPropertyWithValue("connectTimeoutMillis", (int) connectTimeout.toMillis()) + .hasFieldOrPropertyWithValue("callTimeoutMillis", (int) timeout.toMillis()); + }); + } + + @Test + void shouldCustomizeGrpcTransportWhenEnabledWithOtlpGrpcSpanExporterBuilderCustomizer() { + Duration timeout = Duration.ofMinutes(10); + Duration connectTimeout = Duration.ofMinutes(20); + this.contextRunner + .withBean("grpcCustomizer1", OtlpGrpcSpanExporterBuilderCustomizer.class, + () -> (builder) -> builder.setConnectTimeout(connectTimeout)) + .withBean("grpcCustomizer2", OtlpGrpcSpanExporterBuilderCustomizer.class, + () -> (builder) -> builder.setTimeout(timeout)) + .withPropertyValues("management.otlp.tracing.endpoint=http://localhost:4317/v1/traces", + "management.otlp.tracing.transport=grpc") + .run((context) -> { + assertThat(context).hasSingleBean(OtlpGrpcSpanExporter.class).hasSingleBean(SpanExporter.class); + OtlpGrpcSpanExporter exporter = context.getBean(OtlpGrpcSpanExporter.class); + assertThat(exporter).extracting("delegate.grpcSender.client") + .hasFieldOrPropertyWithValue("connectTimeoutMillis", (int) connectTimeout.toMillis()) + .hasFieldOrPropertyWithValue("callTimeoutMillis", (int) timeout.toMillis()); + }); + } + + @Configuration(proxyBeanMethods = false) + private static final class MeterProviderConfiguration { + + static final MeterProvider meterProvider = (instrumentationScopeName) -> null; + + @Bean + MeterProvider meterProvider() { + return meterProvider; + } + + } + @Configuration(proxyBeanMethods = false) private static final class CustomHttpExporterConfiguration { diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/prometheus/LazyTracingSpanContextSupplierTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/prometheus/LazyTracingSpanContextSupplierTests.java deleted file mode 100644 index 17466d234946..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/prometheus/LazyTracingSpanContextSupplierTests.java +++ /dev/null @@ -1,152 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.tracing.prometheus; - -import io.micrometer.tracing.Span; -import io.micrometer.tracing.TraceContext; -import io.micrometer.tracing.Tracer; -import org.junit.jupiter.api.Test; - -import org.springframework.beans.BeansException; -import org.springframework.beans.factory.ObjectProvider; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.mockito.BDDMockito.given; -import static org.mockito.Mockito.mock; - -/** - * Tests for - * {@link org.springframework.boot.actuate.autoconfigure.tracing.prometheus.PrometheusSimpleclientExemplarsAutoConfiguration.LazyTracingSpanContextSupplier}. - * - * @author Andy Wilkinson - */ -@SuppressWarnings({ "deprecation", "removal" }) -class LazyTracingSpanContextSupplierTests { - - private final Tracer tracer = mock(Tracer.class); - - private final ObjectProvider objectProvider = new ObjectProvider<>() { - - @Override - public Tracer getObject() throws BeansException { - return LazyTracingSpanContextSupplierTests.this.tracer; - } - - @Override - public Tracer getObject(Object... args) throws BeansException { - return LazyTracingSpanContextSupplierTests.this.tracer; - } - - @Override - public Tracer getIfAvailable() throws BeansException { - return LazyTracingSpanContextSupplierTests.this.tracer; - } - - @Override - public Tracer getIfUnique() throws BeansException { - return LazyTracingSpanContextSupplierTests.this.tracer; - } - - }; - - private final org.springframework.boot.actuate.autoconfigure.tracing.prometheus.PrometheusSimpleclientExemplarsAutoConfiguration.LazyTracingSpanContextSupplier spanContextSupplier = new org.springframework.boot.actuate.autoconfigure.tracing.prometheus.PrometheusSimpleclientExemplarsAutoConfiguration.LazyTracingSpanContextSupplier( - this.objectProvider); - - @Test - void whenCurrentSpanIsNullThenSpanIdIsNull() { - assertThat(this.spanContextSupplier.getSpanId()).isNull(); - } - - @Test - void whenCurrentSpanIsNullThenTraceIdIsNull() { - assertThat(this.spanContextSupplier.getTraceId()).isNull(); - } - - @Test - void whenCurrentSpanIsNullThenSampledIsFalse() { - assertThat(this.spanContextSupplier.isSampled()).isFalse(); - } - - @Test - void whenCurrentSpanHasSpanIdThenSpanIdIsFromSpan() { - Span span = mock(Span.class); - given(this.tracer.currentSpan()).willReturn(span); - TraceContext traceContext = mock(TraceContext.class); - given(traceContext.spanId()).willReturn("span-id"); - given(span.context()).willReturn(traceContext); - assertThat(this.spanContextSupplier.getSpanId()).isEqualTo("span-id"); - } - - @Test - void whenCurrentSpanHasTraceIdThenTraceIdIsFromSpan() { - Span span = mock(Span.class); - given(this.tracer.currentSpan()).willReturn(span); - TraceContext traceContext = mock(TraceContext.class); - given(traceContext.traceId()).willReturn("trace-id"); - given(span.context()).willReturn(traceContext); - assertThat(this.spanContextSupplier.getTraceId()).isEqualTo("trace-id"); - } - - @Test - void whenCurrentSpanHasNoSpanIdThenSpanIdIsNull() { - Span span = mock(Span.class); - given(this.tracer.currentSpan()).willReturn(span); - TraceContext traceContext = mock(TraceContext.class); - given(span.context()).willReturn(traceContext); - assertThat(this.spanContextSupplier.getSpanId()).isNull(); - } - - @Test - void whenCurrentSpanHasNoTraceIdThenTraceIdIsNull() { - Span span = mock(Span.class); - given(this.tracer.currentSpan()).willReturn(span); - TraceContext traceContext = mock(TraceContext.class); - given(span.context()).willReturn(traceContext); - assertThat(this.spanContextSupplier.getTraceId()).isNull(); - } - - @Test - void whenCurrentSpanIsSampledThenSampledIsTrue() { - Span span = mock(Span.class); - given(this.tracer.currentSpan()).willReturn(span); - TraceContext traceContext = mock(TraceContext.class); - given(traceContext.sampled()).willReturn(true); - given(span.context()).willReturn(traceContext); - assertThat(this.spanContextSupplier.isSampled()).isTrue(); - } - - @Test - void whenCurrentSpanIsNotSampledThenSampledIsFalse() { - Span span = mock(Span.class); - given(this.tracer.currentSpan()).willReturn(span); - TraceContext traceContext = mock(TraceContext.class); - given(traceContext.sampled()).willReturn(false); - given(span.context()).willReturn(traceContext); - assertThat(this.spanContextSupplier.isSampled()).isFalse(); - } - - @Test - void whenCurrentSpanHasDeferredSamplingThenSampledIsFalse() { - Span span = mock(Span.class); - given(this.tracer.currentSpan()).willReturn(span); - TraceContext traceContext = mock(TraceContext.class); - given(traceContext.sampled()).willReturn(null); - given(span.context()).willReturn(traceContext); - assertThat(this.spanContextSupplier.isSampled()).isFalse(); - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/prometheus/PrometheusSimpleclientExemplarsAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/prometheus/PrometheusSimpleclientExemplarsAutoConfigurationTests.java deleted file mode 100644 index b16f687532fa..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/prometheus/PrometheusSimpleclientExemplarsAutoConfigurationTests.java +++ /dev/null @@ -1,135 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.tracing.prometheus; - -import java.util.Optional; -import java.util.regex.Matcher; -import java.util.regex.Pattern; - -import io.micrometer.observation.Observation; -import io.micrometer.observation.ObservationRegistry; -import io.prometheus.client.exemplars.tracer.common.SpanContextSupplier; -import io.prometheus.client.exporter.common.TextFormat; -import org.junit.jupiter.api.Test; - -import org.springframework.boot.actuate.autoconfigure.metrics.export.prometheus.PrometheusSimpleclientMetricsExportAutoConfiguration; -import org.springframework.boot.actuate.autoconfigure.metrics.test.MetricsRun; -import org.springframework.boot.actuate.autoconfigure.observation.ObservationAutoConfiguration; -import org.springframework.boot.actuate.autoconfigure.tracing.BraveAutoConfiguration; -import org.springframework.boot.actuate.autoconfigure.tracing.MicrometerTracingAutoConfiguration; -import org.springframework.boot.autoconfigure.AutoConfigurations; -import org.springframework.boot.test.context.FilteredClassLoader; -import org.springframework.boot.test.context.runner.ApplicationContextRunner; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.util.StringUtils; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.mockito.Mockito.mock; - -/** - * Tests for {@link PrometheusSimpleclientExemplarsAutoConfiguration}. - * - * @author Jonatan Ivanov - */ -@SuppressWarnings("removal") -class PrometheusSimpleclientExemplarsAutoConfigurationTests { - - private static final Pattern BUCKET_TRACE_INFO_PATTERN = Pattern.compile( - "^test_observation_seconds_bucket\\{error=\"none\",le=\".+\"} 1.0 # \\{span_id=\"(\\p{XDigit}+)\",trace_id=\"(\\p{XDigit}+)\"} .+$"); - - private static final Pattern COUNTER_TRACE_INFO_PATTERN = Pattern.compile( - "^test_observation_seconds_count\\{error=\"none\"} 1.0 # \\{span_id=\"(\\p{XDigit}+)\",trace_id=\"(\\p{XDigit}+)\"} .+$"); - - private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() - .withPropertyValues("management.tracing.sampling.probability=1.0", - "management.metrics.distribution.percentiles-histogram.all=true") - .with(MetricsRun.limitedTo()) - .withConfiguration(AutoConfigurations.of(PrometheusSimpleclientMetricsExportAutoConfiguration.class, - PrometheusSimpleclientExemplarsAutoConfiguration.class, ObservationAutoConfiguration.class, - BraveAutoConfiguration.class, MicrometerTracingAutoConfiguration.class)); - - @Test - void shouldNotSupplyBeansIfPrometheusSupportIsMissing() { - this.contextRunner.withClassLoader(new FilteredClassLoader("io.prometheus.client.exemplars")) - .run((context) -> assertThat(context).doesNotHaveBean(SpanContextSupplier.class)); - } - - @Test - void shouldNotSupplyBeansIfMicrometerTracingIsMissing() { - this.contextRunner.withClassLoader(new FilteredClassLoader("io.micrometer.tracing")) - .run((context) -> assertThat(context).doesNotHaveBean(SpanContextSupplier.class)); - } - - @Test - void shouldSupplyCustomBeans() { - this.contextRunner.withUserConfiguration(CustomConfiguration.class) - .run((context) -> assertThat(context).hasSingleBean(SpanContextSupplier.class) - .getBean(SpanContextSupplier.class) - .isSameAs(CustomConfiguration.SUPPLIER)); - } - - @Test - @SuppressWarnings("deprecation") - void prometheusOpenMetricsOutputShouldContainExemplars() { - this.contextRunner.run((context) -> { - assertThat(context).hasSingleBean(SpanContextSupplier.class); - ObservationRegistry observationRegistry = context.getBean(ObservationRegistry.class); - Observation.start("test.observation", observationRegistry).stop(); - io.micrometer.prometheus.PrometheusMeterRegistry prometheusMeterRegistry = context - .getBean(io.micrometer.prometheus.PrometheusMeterRegistry.class); - String openMetricsOutput = prometheusMeterRegistry.scrape(TextFormat.CONTENT_TYPE_OPENMETRICS_100); - - assertThat(openMetricsOutput).contains("test_observation_seconds_bucket"); - assertThat(openMetricsOutput).containsOnlyOnce("test_observation_seconds_count"); - assertThat(StringUtils.countOccurrencesOf(openMetricsOutput, "span_id")).isEqualTo(2); - assertThat(StringUtils.countOccurrencesOf(openMetricsOutput, "trace_id")).isEqualTo(2); - - Optional bucketTraceInfo = openMetricsOutput.lines() - .filter((line) -> line.contains("test_observation_seconds_bucket") && line.contains("span_id")) - .map(BUCKET_TRACE_INFO_PATTERN::matcher) - .flatMap(Matcher::results) - .map((matchResult) -> new TraceInfo(matchResult.group(2), matchResult.group(1))) - .findFirst(); - - Optional counterTraceInfo = openMetricsOutput.lines() - .filter((line) -> line.contains("test_observation_seconds_count") && line.contains("span_id")) - .map(COUNTER_TRACE_INFO_PATTERN::matcher) - .flatMap(Matcher::results) - .map((matchResult) -> new TraceInfo(matchResult.group(2), matchResult.group(1))) - .findFirst(); - - assertThat(bucketTraceInfo).isNotEmpty().contains(counterTraceInfo.orElse(null)); - }); - } - - @Configuration(proxyBeanMethods = false) - private static final class CustomConfiguration { - - static final SpanContextSupplier SUPPLIER = mock(SpanContextSupplier.class); - - @Bean - SpanContextSupplier customSpanContextSupplier() { - return SUPPLIER; - } - - } - - private record TraceInfo(String traceId, String spanId) { - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinAutoConfigurationIntegrationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinAutoConfigurationIntegrationTests.java index b47588d2d3d8..14b92a967bd5 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinAutoConfigurationIntegrationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinAutoConfigurationIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,7 +17,6 @@ package org.springframework.boot.actuate.autoconfigure.tracing.zipkin; import org.junit.jupiter.api.Test; -import zipkin2.reporter.urlconnection.URLConnectionSender; import org.springframework.boot.actuate.autoconfigure.metrics.MetricsAutoConfiguration; import org.springframework.boot.actuate.autoconfigure.metrics.export.simple.SimpleMetricsExportAutoConfiguration; @@ -28,7 +27,6 @@ import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.web.client.RestTemplateAutoConfiguration; import org.springframework.boot.autoconfigure.web.reactive.function.client.WebClientAutoConfiguration; -import org.springframework.boot.test.context.FilteredClassLoader; import org.springframework.boot.test.context.assertj.ApplicationContextAssertProvider; import org.springframework.boot.test.context.runner.AbstractApplicationContextRunner; import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner; @@ -61,12 +59,10 @@ void zipkinsUseOfWebClientDoesNotCauseACycle() { , C extends ConfigurableApplicationContext, A extends ApplicationContextAssertProvider> AbstractApplicationContextRunner configure( AbstractApplicationContextRunner runner) { - return runner - .withConfiguration(AutoConfigurations.of(MicrometerTracingAutoConfiguration.class, - ObservationAutoConfiguration.class, BraveAutoConfiguration.class, ZipkinAutoConfiguration.class, - HttpClientObservationsAutoConfiguration.class, MetricsAutoConfiguration.class, - SimpleMetricsExportAutoConfiguration.class)) - .withClassLoader(new FilteredClassLoader(URLConnectionSender.class)); + return runner.withConfiguration(AutoConfigurations.of(MicrometerTracingAutoConfiguration.class, + ObservationAutoConfiguration.class, BraveAutoConfiguration.class, ZipkinAutoConfiguration.class, + HttpClientObservationsAutoConfiguration.class, MetricsAutoConfiguration.class, + SimpleMetricsExportAutoConfiguration.class)); } } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinAutoConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinAutoConfigurationTests.java index 3a5323cef723..b568900accc3 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -73,7 +73,7 @@ void shouldUseCustomConnectionDetailsWhenDefined() { @Test void shouldWorkWithoutSenders() { this.contextRunner - .withClassLoader(new FilteredClassLoader("zipkin2.reporter.urlconnection", "org.springframework.web.client", + .withClassLoader(new FilteredClassLoader("org.springframework.web.client", "org.springframework.web.reactive.function.client")) .run((context) -> assertThat(context).hasNotFailed()); } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinConfigurationsSenderConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinConfigurationsSenderConfigurationTests.java index d5c2ddee7382..4767ecc2091a 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinConfigurationsSenderConfigurationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinConfigurationsSenderConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,150 +16,47 @@ package org.springframework.boot.actuate.autoconfigure.tracing.zipkin; -import java.io.IOException; -import java.nio.charset.StandardCharsets; -import java.util.List; -import java.util.concurrent.TimeUnit; +import java.net.http.HttpClient; -import okhttp3.mockwebserver.MockResponse; -import okhttp3.mockwebserver.MockWebServer; -import okhttp3.mockwebserver.RecordedRequest; import org.junit.jupiter.api.Test; -import org.mockito.ArgumentMatchers; import zipkin2.reporter.BytesMessageSender; import zipkin2.reporter.HttpEndpointSupplier; -import zipkin2.reporter.urlconnection.URLConnectionSender; +import org.springframework.boot.actuate.autoconfigure.tracing.zipkin.ZipkinConfigurations.HttpClientSenderConfiguration; import org.springframework.boot.actuate.autoconfigure.tracing.zipkin.ZipkinConfigurations.SenderConfiguration; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.test.context.FilteredClassLoader; import org.springframework.boot.test.context.runner.ApplicationContextRunner; -import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner; -import org.springframework.boot.test.context.runner.WebApplicationContextRunner; -import org.springframework.boot.web.client.RestTemplateBuilder; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; -import org.springframework.web.reactive.function.client.WebClient; import static org.assertj.core.api.Assertions.assertThat; -import static org.mockito.BDDMockito.then; import static org.mockito.Mockito.mock; /** * Tests for {@link SenderConfiguration}. * * @author Moritz Halbritter + * @author Wick Dynex */ -@SuppressWarnings({ "deprecation", "removal" }) class ZipkinConfigurationsSenderConfigurationTests { private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() .withConfiguration(AutoConfigurations.of(DefaultEncodingConfiguration.class, SenderConfiguration.class)); - private final ReactiveWebApplicationContextRunner reactiveContextRunner = new ReactiveWebApplicationContextRunner() - .withConfiguration(AutoConfigurations.of(DefaultEncodingConfiguration.class, SenderConfiguration.class)); - - private final WebApplicationContextRunner servletContextRunner = new WebApplicationContextRunner() - .withConfiguration(AutoConfigurations.of(DefaultEncodingConfiguration.class, SenderConfiguration.class)); - @Test - void shouldSupplyBeans() { + void shouldSupplyDefaultHttpClientSenderBean() { this.contextRunner.run((context) -> { assertThat(context).hasSingleBean(BytesMessageSender.class); - assertThat(context).hasSingleBean(URLConnectionSender.class); - assertThat(context).doesNotHaveBean(ZipkinRestTemplateSender.class); + assertThat(context).hasSingleBean(ZipkinHttpClientSender.class); }); } @Test - void shouldUseHttpClientIfUrlSenderIsNotAvailable() { - this.contextRunner.withUserConfiguration(HttpClientConfiguration.class) - .withClassLoader(new FilteredClassLoader("zipkin2.reporter.urlconnection", "org.springframework.web.client", - "org.springframework.web.reactive.function.client")) - .run((context) -> { - assertThat(context).doesNotHaveBean(URLConnectionSender.class); - assertThat(context).hasSingleBean(BytesMessageSender.class); - assertThat(context).hasSingleBean(ZipkinHttpClientSender.class); - then(context.getBean(ZipkinHttpClientBuilderCustomizer.class)).should() - .customize(ArgumentMatchers.any()); - }); - } - - @Test - void shouldPreferWebClientSenderIfWebApplicationIsReactiveAndUrlSenderIsNotAvailable() { - this.reactiveContextRunner.withUserConfiguration(RestTemplateConfiguration.class, WebClientConfiguration.class) - .withClassLoader(new FilteredClassLoader("zipkin2.reporter.urlconnection")) - .run((context) -> { - assertThat(context).doesNotHaveBean(URLConnectionSender.class); - assertThat(context).hasSingleBean(BytesMessageSender.class); - assertThat(context).hasSingleBean(ZipkinWebClientSender.class); - then(context.getBean(ZipkinWebClientBuilderCustomizer.class)).should() - .customize(ArgumentMatchers.any()); - }); - } - - @Test - void shouldPreferWebClientSenderIfWebApplicationIsServletAndUrlSenderIsNotAvailable() { - this.servletContextRunner.withUserConfiguration(RestTemplateConfiguration.class, WebClientConfiguration.class) - .withClassLoader(new FilteredClassLoader("zipkin2.reporter.urlconnection")) - .run((context) -> { - assertThat(context).doesNotHaveBean(URLConnectionSender.class); - assertThat(context).hasSingleBean(BytesMessageSender.class); - assertThat(context).hasSingleBean(ZipkinWebClientSender.class); - }); - } - - @Test - void shouldPreferWebClientInNonWebApplicationAndUrlConnectionSenderIsNotAvailable() { - this.contextRunner.withUserConfiguration(RestTemplateConfiguration.class, WebClientConfiguration.class) - .withClassLoader(new FilteredClassLoader("zipkin2.reporter.urlconnection")) - .run((context) -> { - assertThat(context).doesNotHaveBean(URLConnectionSender.class); - assertThat(context).hasSingleBean(BytesMessageSender.class); - assertThat(context).hasSingleBean(ZipkinWebClientSender.class); - }); - } - - @Test - void willUseRestTemplateInNonWebApplicationIfUrlConnectionSenderAndWebClientAreNotAvailable() { - this.contextRunner.withUserConfiguration(RestTemplateConfiguration.class) - .withClassLoader(new FilteredClassLoader(URLConnectionSender.class, WebClient.class)) - .run((context) -> { - assertThat(context).doesNotHaveBean(URLConnectionSender.class); - assertThat(context).hasSingleBean(BytesMessageSender.class); - assertThat(context).hasSingleBean(ZipkinRestTemplateSender.class); - }); - } - - @Test - void willUseRestTemplateInServletWebApplicationIfUrlConnectionSenderAndWebClientNotAvailable() { - this.servletContextRunner.withUserConfiguration(RestTemplateConfiguration.class) - .withClassLoader(new FilteredClassLoader(URLConnectionSender.class, WebClient.class)) - .run((context) -> { - assertThat(context).doesNotHaveBean(URLConnectionSender.class); - assertThat(context).hasSingleBean(BytesMessageSender.class); - assertThat(context).hasSingleBean(ZipkinRestTemplateSender.class); - }); - } - - @Test - void willUseRestTemplateInReactiveWebApplicationIfUrlConnectionSenderAndWebClientAreNotAvailable() { - this.reactiveContextRunner.withUserConfiguration(RestTemplateConfiguration.class) - .withClassLoader(new FilteredClassLoader(URLConnectionSender.class, WebClient.class)) - .run((context) -> { - assertThat(context).doesNotHaveBean(URLConnectionSender.class); - assertThat(context).hasSingleBean(BytesMessageSender.class); - assertThat(context).hasSingleBean(ZipkinRestTemplateSender.class); - }); - } - - @Test - void shouldNotUseWebClientSenderIfNoBuilderIsAvailable() { - this.reactiveContextRunner.run((context) -> { - assertThat(context).doesNotHaveBean(ZipkinWebClientSender.class); - assertThat(context).hasSingleBean(BytesMessageSender.class); - assertThat(context).hasSingleBean(URLConnectionSender.class); - }); + void shouldNotProvideHttpClientSenderIfHttpClientIsNotAvailable() { + this.contextRunner.withUserConfiguration(HttpClientSenderConfiguration.class) + .withClassLoader(new FilteredClassLoader(HttpClient.class)) + .run((context) -> assertThat(context).doesNotHaveBean(ZipkinHttpClientSender.class)); } @Test @@ -170,85 +67,18 @@ void shouldBackOffOnCustomBeans() { }); } - @Test - void shouldApplyZipkinRestTemplateBuilderCustomizers() throws IOException { - try (MockWebServer mockWebServer = new MockWebServer()) { - mockWebServer.enqueue(new MockResponse().setResponseCode(204)); - this.reactiveContextRunner - .withPropertyValues("management.zipkin.tracing.endpoint=" + mockWebServer.url("/")) - .withUserConfiguration(RestTemplateConfiguration.class) - .withClassLoader(new FilteredClassLoader(URLConnectionSender.class, WebClient.class)) - .run((context) -> { - assertThat(context).hasSingleBean(ZipkinRestTemplateSender.class); - ZipkinRestTemplateSender sender = context.getBean(ZipkinRestTemplateSender.class); - sender.send(List.of("spans".getBytes(StandardCharsets.UTF_8))); - RecordedRequest recordedRequest = mockWebServer.takeRequest(1, TimeUnit.SECONDS); - assertThat(recordedRequest).isNotNull(); - assertThat(recordedRequest.getHeaders().get("x-dummy")).isEqualTo("dummy"); - }); - } - } - @Test void shouldUseCustomHttpEndpointSupplierFactory() { this.contextRunner.withUserConfiguration(CustomHttpEndpointSupplierFactoryConfiguration.class) - .run((context) -> assertThat(context.getBean(URLConnectionSender.class)) - .extracting("delegate.endpointSupplier") - .isInstanceOf(CustomHttpEndpointSupplier.class)); - } - - @Test - @SuppressWarnings("resource") - void shouldUseCustomHttpEndpointSupplierFactoryWhenReactive() { - this.reactiveContextRunner.withUserConfiguration(WebClientConfiguration.class) - .withClassLoader(new FilteredClassLoader(URLConnectionSender.class)) - .withUserConfiguration(CustomHttpEndpointSupplierFactoryConfiguration.class) - .run((context) -> assertThat(context.getBean(ZipkinWebClientSender.class)).extracting("endpointSupplier") - .isInstanceOf(CustomHttpEndpointSupplier.class)); - } - - @Test - @SuppressWarnings("resource") - void shouldUseCustomHttpEndpointSupplierFactoryWhenRestTemplate() { - this.contextRunner.withUserConfiguration(RestTemplateConfiguration.class) - .withClassLoader(new FilteredClassLoader(URLConnectionSender.class, WebClient.class)) - .withUserConfiguration(CustomHttpEndpointSupplierFactoryConfiguration.class) - .run((context) -> assertThat(context.getBean(ZipkinRestTemplateSender.class)).extracting("endpointSupplier") - .isInstanceOf(CustomHttpEndpointSupplier.class)); - } - - @Configuration(proxyBeanMethods = false) - private static final class RestTemplateConfiguration { - - @Bean - ZipkinRestTemplateBuilderCustomizer zipkinRestTemplateBuilderCustomizer() { - return new DummyZipkinRestTemplateBuilderCustomizer(); - } - - } - - @Configuration(proxyBeanMethods = false) - private static final class WebClientConfiguration { - - @Bean - ZipkinWebClientBuilderCustomizer webClientBuilder() { - return mock(ZipkinWebClientBuilderCustomizer.class); - } - - } - - @Configuration(proxyBeanMethods = false) - private static final class HttpClientConfiguration { - - @Bean - ZipkinHttpClientBuilderCustomizer httpClientBuilderCustomizer() { - return mock(ZipkinHttpClientBuilderCustomizer.class); - } - + .run((context) -> { + ZipkinHttpClientSender httpClientSender = context.getBean(ZipkinHttpClientSender.class); + assertThat(httpClientSender).extracting("endpointSupplier") + .isInstanceOf(CustomHttpEndpointSupplier.class); + }); } @Configuration(proxyBeanMethods = false) - private static final class CustomConfiguration { + static class CustomConfiguration { @Bean BytesMessageSender customSender() { @@ -257,17 +87,8 @@ BytesMessageSender customSender() { } - private static final class DummyZipkinRestTemplateBuilderCustomizer implements ZipkinRestTemplateBuilderCustomizer { - - @Override - public RestTemplateBuilder customize(RestTemplateBuilder restTemplateBuilder) { - return restTemplateBuilder.defaultHeader("x-dummy", "dummy"); - } - - } - @Configuration(proxyBeanMethods = false) - private static final class CustomHttpEndpointSupplierFactoryConfiguration { + static class CustomHttpEndpointSupplierFactoryConfiguration { @Bean HttpEndpointSupplier.Factory httpEndpointSupplier() { @@ -276,7 +97,7 @@ HttpEndpointSupplier.Factory httpEndpointSupplier() { } - private static final class CustomHttpEndpointSupplierFactory implements HttpEndpointSupplier.Factory { + static class CustomHttpEndpointSupplierFactory implements HttpEndpointSupplier.Factory { @Override public HttpEndpointSupplier create(String endpoint) { @@ -285,7 +106,13 @@ public HttpEndpointSupplier create(String endpoint) { } - private record CustomHttpEndpointSupplier(String endpoint) implements HttpEndpointSupplier { + static class CustomHttpEndpointSupplier implements HttpEndpointSupplier { + + private final String endpoint; + + CustomHttpEndpointSupplier(String endpoint) { + this.endpoint = endpoint; + } @Override public String get() { @@ -295,6 +122,7 @@ public String get() { @Override public void close() { } + } } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinRestTemplateSenderTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinRestTemplateSenderTests.java deleted file mode 100644 index f753677acc90..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinRestTemplateSenderTests.java +++ /dev/null @@ -1,139 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.tracing.zipkin; - -import java.io.IOException; -import java.util.Base64; -import java.util.Collections; -import java.util.List; - -import org.junit.jupiter.api.AfterEach; -import org.junit.jupiter.api.Test; -import zipkin2.reporter.BytesMessageSender; -import zipkin2.reporter.Encoding; -import zipkin2.reporter.HttpEndpointSupplier; -import zipkin2.reporter.HttpEndpointSuppliers; - -import org.springframework.http.HttpMethod; -import org.springframework.http.HttpStatus; -import org.springframework.test.web.client.MockRestServiceServer; -import org.springframework.web.client.RestTemplate; - -import static org.assertj.core.api.Assertions.assertThatException; -import static org.springframework.test.web.client.match.MockRestRequestMatchers.content; -import static org.springframework.test.web.client.match.MockRestRequestMatchers.header; -import static org.springframework.test.web.client.match.MockRestRequestMatchers.method; -import static org.springframework.test.web.client.match.MockRestRequestMatchers.requestTo; -import static org.springframework.test.web.client.response.MockRestResponseCreators.withStatus; - -/** - * Tests for {@link ZipkinRestTemplateSender}. - * - * @author Moritz Halbritter - * @author Stefan Bratanov - */ -@SuppressWarnings({ "deprecation", "removal" }) -class ZipkinRestTemplateSenderTests extends ZipkinHttpSenderTests { - - private static final String ZIPKIN_URL = "http://localhost:9411/api/v2/spans"; - - private RestTemplate restTemplate; - - private MockRestServiceServer mockServer; - - @Override - BytesMessageSender createSender() { - this.restTemplate = new RestTemplate(); - this.mockServer = MockRestServiceServer.createServer(this.restTemplate); - return createSender(Encoding.JSON); - } - - BytesMessageSender createSender(Encoding encoding) { - return createSender(HttpEndpointSuppliers.constantFactory(), encoding); - } - - BytesMessageSender createSender(HttpEndpointSupplier.Factory endpointSupplierFactory, Encoding encoding) { - return new ZipkinRestTemplateSender(encoding, endpointSupplierFactory, ZIPKIN_URL, this.restTemplate); - } - - @AfterEach - @Override - void afterEach() throws IOException { - super.afterEach(); - this.mockServer.verify(); - } - - @Test - void sendShouldSendSpansToZipkin() throws IOException { - this.mockServer.expect(requestTo(ZIPKIN_URL)) - .andExpect(method(HttpMethod.POST)) - .andExpect(content().contentType("application/json")) - .andExpect(content().string("[span1,span2]")) - .andRespond(withStatus(HttpStatus.ACCEPTED)); - this.sender.send(List.of(toByteArray("span1"), toByteArray("span2"))); - } - - @Test - void sendShouldSendSpansToZipkinInProto3() throws IOException { - this.mockServer.expect(requestTo(ZIPKIN_URL)) - .andExpect(method(HttpMethod.POST)) - .andExpect(content().contentType("application/x-protobuf")) - .andExpect(content().string("span1span2")) - .andRespond(withStatus(HttpStatus.ACCEPTED)); - try (BytesMessageSender sender = createSender(Encoding.PROTO3)) { - sender.send(List.of(toByteArray("span1"), toByteArray("span2"))); - } - } - - @Test - void sendUsesDynamicEndpoint() throws Exception { - this.mockServer.expect(requestTo(ZIPKIN_URL + "/1")).andRespond(withStatus(HttpStatus.ACCEPTED)); - this.mockServer.expect(requestTo(ZIPKIN_URL + "/2")).andRespond(withStatus(HttpStatus.ACCEPTED)); - try (HttpEndpointSupplier httpEndpointSupplier = new TestHttpEndpointSupplier(ZIPKIN_URL)) { - try (BytesMessageSender sender = createSender((endpoint) -> httpEndpointSupplier, Encoding.JSON)) { - sender.send(Collections.emptyList()); - sender.send(Collections.emptyList()); - } - } - } - - @Test - void sendShouldHandleHttpFailures() { - this.mockServer.expect(requestTo(ZIPKIN_URL)) - .andExpect(method(HttpMethod.POST)) - .andRespond(withStatus(HttpStatus.INTERNAL_SERVER_ERROR)); - - assertThatException().isThrownBy(() -> this.sender.send(Collections.emptyList())) - .withMessageContaining("500 Internal Server Error"); - } - - @Test - void sendShouldCompressData() throws IOException { - String uncompressed = "a".repeat(10000); - // This is gzip compressed 10000 times 'a' - byte[] compressed = Base64.getDecoder() - .decode("H4sIAAAAAAAA/+3BMQ0AAAwDIKFLj/k3UR8NcA8AAAAAAAAAAAADUsAZfeASJwAA"); - this.mockServer.expect(requestTo(ZIPKIN_URL)) - .andExpect(method(HttpMethod.POST)) - .andExpect(header("Content-Encoding", "gzip")) - .andExpect(content().contentType("application/json")) - .andExpect(content().bytes(compressed)) - .andRespond(withStatus(HttpStatus.ACCEPTED)); - this.sender.send(List.of(toByteArray(uncompressed))); - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinWebClientSenderTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinWebClientSenderTests.java deleted file mode 100644 index 71ad42aa5457..000000000000 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/tracing/zipkin/ZipkinWebClientSenderTests.java +++ /dev/null @@ -1,204 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.autoconfigure.tracing.zipkin; - -import java.io.IOException; -import java.time.Duration; -import java.util.Base64; -import java.util.Collections; -import java.util.List; -import java.util.concurrent.TimeUnit; -import java.util.concurrent.TimeoutException; -import java.util.function.Consumer; - -import okhttp3.mockwebserver.MockResponse; -import okhttp3.mockwebserver.MockWebServer; -import okhttp3.mockwebserver.QueueDispatcher; -import okhttp3.mockwebserver.RecordedRequest; -import org.junit.jupiter.api.AfterAll; -import org.junit.jupiter.api.BeforeAll; -import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Test; -import zipkin2.reporter.BytesMessageSender; -import zipkin2.reporter.Encoding; -import zipkin2.reporter.HttpEndpointSupplier; -import zipkin2.reporter.HttpEndpointSuppliers; - -import org.springframework.web.reactive.function.client.WebClient; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.assertThatException; - -/** - * Tests for {@link ZipkinWebClientSender}. - * - * @author Stefan Bratanov - */ -@SuppressWarnings({ "deprecation", "removal" }) -class ZipkinWebClientSenderTests extends ZipkinHttpSenderTests { - - private static final Duration TIMEOUT = Duration.ofSeconds(30); - - private static ClearableDispatcher dispatcher; - - private static MockWebServer mockBackEnd; - - private static String ZIPKIN_URL; - - @BeforeAll - static void beforeAll() throws IOException { - dispatcher = new ClearableDispatcher(); - mockBackEnd = new MockWebServer(); - mockBackEnd.setDispatcher(dispatcher); - mockBackEnd.start(); - ZIPKIN_URL = mockBackEnd.url("/api/v2/spans").toString(); - } - - @AfterAll - static void afterAll() throws IOException { - mockBackEnd.shutdown(); - } - - @Override - @BeforeEach - void beforeEach() { - super.beforeEach(); - clearResponses(); - clearRequests(); - } - - @Override - BytesMessageSender createSender() { - return createSender(Encoding.JSON, TIMEOUT); - } - - ZipkinWebClientSender createSender(Encoding encoding, Duration timeout) { - return createSender(HttpEndpointSuppliers.constantFactory(), encoding, timeout); - } - - ZipkinWebClientSender createSender(HttpEndpointSupplier.Factory endpointSupplierFactory, Encoding encoding, - Duration timeout) { - WebClient webClient = WebClient.builder().build(); - return new ZipkinWebClientSender(encoding, endpointSupplierFactory, ZIPKIN_URL, webClient, timeout); - } - - @Test - void sendShouldSendSpansToZipkin() throws IOException, InterruptedException { - mockBackEnd.enqueue(new MockResponse()); - List encodedSpans = List.of(toByteArray("span1"), toByteArray("span2")); - this.sender.send(encodedSpans); - requestAssertions((request) -> { - assertThat(request.getMethod()).isEqualTo("POST"); - assertThat(request.getHeader("Content-Type")).isEqualTo("application/json"); - assertThat(request.getBody().readUtf8()).isEqualTo("[span1,span2]"); - }); - } - - @Test - void sendShouldSendSpansToZipkinInProto3() throws IOException, InterruptedException { - mockBackEnd.enqueue(new MockResponse()); - List encodedSpans = List.of(toByteArray("span1"), toByteArray("span2")); - try (BytesMessageSender sender = createSender(Encoding.PROTO3, TIMEOUT)) { - sender.send(encodedSpans); - } - requestAssertions((request) -> { - assertThat(request.getMethod()).isEqualTo("POST"); - assertThat(request.getHeader("Content-Type")).isEqualTo("application/x-protobuf"); - assertThat(request.getBody().readUtf8()).isEqualTo("span1span2"); - }); - } - - @Test - void sendUsesDynamicEndpoint() throws Exception { - mockBackEnd.enqueue(new MockResponse()); - mockBackEnd.enqueue(new MockResponse()); - try (HttpEndpointSupplier httpEndpointSupplier = new TestHttpEndpointSupplier(ZIPKIN_URL)) { - try (BytesMessageSender sender = createSender((endpoint) -> httpEndpointSupplier, Encoding.JSON, TIMEOUT)) { - sender.send(Collections.emptyList()); - sender.send(Collections.emptyList()); - } - assertThat(mockBackEnd.takeRequest().getPath()).endsWith("/1"); - assertThat(mockBackEnd.takeRequest().getPath()).endsWith("/2"); - } - } - - @Test - void sendShouldHandleHttpFailures() throws InterruptedException { - mockBackEnd.enqueue(new MockResponse().setResponseCode(500)); - assertThatException().isThrownBy(() -> this.sender.send(Collections.emptyList())) - .withMessageContaining("500 Internal Server Error"); - requestAssertions((request) -> assertThat(request.getMethod()).isEqualTo("POST")); - } - - @Test - void sendShouldCompressData() throws IOException, InterruptedException { - String uncompressed = "a".repeat(10000); - // This is gzip compressed 10000 times 'a' - byte[] compressed = Base64.getDecoder() - .decode("H4sIAAAAAAAA/+3BMQ0AAAwDIKFLj/k3UR8NcA8AAAAAAAAAAAADUsAZfeASJwAA"); - mockBackEnd.enqueue(new MockResponse()); - this.sender.send(List.of(toByteArray(uncompressed))); - requestAssertions((request) -> { - assertThat(request.getMethod()).isEqualTo("POST"); - assertThat(request.getHeader("Content-Type")).isEqualTo("application/json"); - assertThat(request.getHeader("Content-Encoding")).isEqualTo("gzip"); - assertThat(request.getBody().readByteArray()).isEqualTo(compressed); - }); - } - - @Test - void shouldTimeout() throws IOException { - try (BytesMessageSender sender = createSender(Encoding.JSON, Duration.ofMillis(1))) { - MockResponse response = new MockResponse().setResponseCode(200).setHeadersDelay(100, TimeUnit.MILLISECONDS); - mockBackEnd.enqueue(response); - assertThatException().isThrownBy(() -> sender.send(Collections.emptyList())) - .withCauseInstanceOf(TimeoutException.class); - } - } - - private void requestAssertions(Consumer assertions) throws InterruptedException { - RecordedRequest request = mockBackEnd.takeRequest(); - assertThat(request).satisfies(assertions); - } - - private static void clearRequests() { - RecordedRequest request; - do { - try { - request = mockBackEnd.takeRequest(0, TimeUnit.SECONDS); - } - catch (InterruptedException ex) { - Thread.currentThread().interrupt(); - throw new RuntimeException(ex); - } - } - while (request != null); - } - - private static void clearResponses() { - dispatcher.clear(); - } - - private static final class ClearableDispatcher extends QueueDispatcher { - - void clear() { - getResponseQueue().clear(); - } - - } - -} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/wavefront/WavefrontPropertiesTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/wavefront/WavefrontPropertiesTests.java index 719861b5da85..403a949f595e 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/wavefront/WavefrontPropertiesTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/wavefront/WavefrontPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/mappings/MappingsEndpointServletDocumentationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/mappings/MappingsEndpointServletDocumentationTests.java index c18c15fb322f..08797a7fe1d4 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/mappings/MappingsEndpointServletDocumentationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/mappings/MappingsEndpointServletDocumentationTests.java @@ -47,6 +47,9 @@ import org.springframework.test.web.reactive.server.WebTestClient; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.servlet.function.RouterFunction; +import org.springframework.web.servlet.function.RouterFunctions; +import org.springframework.web.servlet.function.ServerResponse; import static org.springframework.restdocs.payload.PayloadDocumentation.beneathPath; import static org.springframework.restdocs.payload.PayloadDocumentation.fieldWithPath; @@ -54,11 +57,13 @@ import static org.springframework.restdocs.payload.PayloadDocumentation.subsectionWithPath; import static org.springframework.restdocs.webtestclient.WebTestClientRestDocumentation.document; import static org.springframework.restdocs.webtestclient.WebTestClientRestDocumentation.documentationConfiguration; +import static org.springframework.web.servlet.function.RequestPredicates.GET; /** * Tests for generating documentation describing {@link MappingsEndpoint}. * * @author Andy Wilkinson + * @author Xiong Tang */ @ExtendWith(RestDocumentationExtension.class) @SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT) @@ -130,6 +135,13 @@ void mappings() { fieldWithPath("*.[].details.handlerMethod.name").description("Name of the method."), fieldWithPath("*.[].details.handlerMethod.descriptor") .description("Descriptor of the method as specified in the Java Language Specification.")); + List handlerFunction = List.of( + fieldWithPath("*.[].details.handlerFunction").optional() + .type(JsonFieldType.OBJECT) + .description("Details of the function, if any, that will handle requests to this mapping."), + fieldWithPath("*.[].details.handlerFunction.className").type(JsonFieldType.STRING) + .description("Fully qualified name of the class of the function.")); + dispatcherServletFields.addAll(handlerFunction); dispatcherServletFields.addAll(handlerMethod); dispatcherServletFields.addAll(requestMappingConditions); this.client.get() @@ -191,6 +203,11 @@ ExampleController exampleController() { return new ExampleController(); } + @Bean + RouterFunction exampleRouter() { + return RouterFunctions.route(GET("/foo"), (request) -> ServerResponse.ok().build()); + } + } @RestController diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfigurationIntegrationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfigurationIntegrationTests.java index 3f656f517c8c..94b1896ee3cf 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfigurationIntegrationTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfigurationIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,8 +19,13 @@ import java.io.IOException; import java.nio.charset.StandardCharsets; import java.nio.file.Path; +import java.util.ArrayList; +import java.util.List; import java.util.function.Consumer; +import org.apache.catalina.Valve; +import org.apache.catalina.startup.Tomcat; +import org.apache.catalina.valves.AccessLogValve; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.io.TempDir; @@ -39,7 +44,11 @@ import org.springframework.boot.test.context.runner.ContextConsumer; import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner; import org.springframework.boot.web.context.ServerPortInfoApplicationContextInitializer; +import org.springframework.boot.web.context.WebServerInitializedEvent; +import org.springframework.boot.web.embedded.tomcat.TomcatWebServer; import org.springframework.boot.web.reactive.context.AnnotationConfigReactiveWebServerApplicationContext; +import org.springframework.boot.web.server.WebServer; +import org.springframework.context.ApplicationListener; import org.springframework.context.ConfigurableApplicationContext; import org.springframework.core.convert.support.ConfigurableConversionService; import org.springframework.http.MediaType; @@ -55,6 +64,8 @@ */ class ReactiveManagementChildContextConfigurationIntegrationTests { + private final List webServers = new ArrayList<>(); + private final ReactiveWebApplicationContextRunner runner = new ReactiveWebApplicationContextRunner( AnnotationConfigReactiveWebServerApplicationContext::new) .withConfiguration(AutoConfigurations.of(ManagementContextAutoConfiguration.class, @@ -63,6 +74,8 @@ class ReactiveManagementChildContextConfigurationIntegrationTests { WebFluxAutoConfiguration.class)) .withUserConfiguration(SucceedingEndpoint.class) .withInitializer(new ServerPortInfoApplicationContextInitializer()) + .withInitializer((context) -> context.addApplicationListener( + (ApplicationListener) (event) -> this.webServers.add(event.getWebServer()))) .withPropertyValues("server.port=0", "management.server.port=0", "management.endpoints.web.exposure.include=*"); @TempDir @@ -99,6 +112,26 @@ void whenManagementServerPortLoadedFromConfigTree() { .run((context) -> assertThat(context).hasNotFailed()); } + @Test + void accessLogHasManagementServerSpecificPrefix() { + this.runner.withPropertyValues("server.tomcat.accesslog.enabled=true").run((context) -> { + AccessLogValve accessLogValve = findAccessLogValve(); + assertThat(accessLogValve).isNotNull(); + assertThat(accessLogValve.getPrefix()).isEqualTo("management_access_log"); + }); + } + + private AccessLogValve findAccessLogValve() { + assertThat(this.webServers).hasSize(2); + Tomcat tomcat = ((TomcatWebServer) this.webServers.get(1)).getTomcat(); + for (Valve valve : tomcat.getEngine().getPipeline().getValves()) { + if (valve instanceof AccessLogValve accessLogValve) { + return accessLogValve; + } + } + return null; + } + private void addConfigTreePropertySource(ConfigurableApplicationContext applicationContext) { try { applicationContext.getEnvironment() diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfigurationTests.java new file mode 100644 index 000000000000..7bfbd7153a7d --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/reactive/ReactiveManagementChildContextConfigurationTests.java @@ -0,0 +1,49 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.web.reactive; + +import org.junit.jupiter.api.Test; + +import org.springframework.boot.actuate.autoconfigure.web.reactive.ReactiveManagementChildContextConfiguration.AccessLogCustomizer; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link ReactiveManagementChildContextConfiguration}. + * + * @author Moritz Halbritter + */ +class ReactiveManagementChildContextConfigurationTests { + + @Test + void accessLogCustomizer() { + AccessLogCustomizer customizer = new AccessLogCustomizer("prefix") { + }; + assertThat(customizer.customizePrefix(null)).isEqualTo("prefix"); + assertThat(customizer.customizePrefix("existing")).isEqualTo("prefixexisting"); + assertThat(customizer.customizePrefix("prefixexisting")).isEqualTo("prefixexisting"); + } + + @Test + void accessLogCustomizerWithNullPrefix() { + AccessLogCustomizer customizer = new AccessLogCustomizer(null) { + }; + assertThat(customizer.customizePrefix(null)).isEqualTo(null); + assertThat(customizer.customizePrefix("existing")).isEqualTo("existing"); + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementServerPropertiesTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementServerPropertiesTests.java index 8ab33edfe81b..46edd410c234 100644 --- a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementServerPropertiesTests.java +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/server/ManagementServerPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,6 +25,7 @@ * * @author Phillip Webb * @author Stephane Nicoll + * @author Moritz Halbritter */ class ManagementServerPropertiesTests { @@ -68,4 +69,12 @@ void slashOfBasePathIsDefaultValue() { assertThat(properties.getBasePath()).isEmpty(); } + @Test + void accessLogsArePrefixedByDefault() { + ManagementServerProperties properties = new ManagementServerProperties(); + assertThat(properties.getTomcat().getAccesslog().getPrefix()).isEqualTo("management_"); + assertThat(properties.getJetty().getAccesslog().getPrefix()).isEqualTo("management_"); + assertThat(properties.getUndertow().getAccesslog().getPrefix()).isEqualTo("management_"); + } + } diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementChildContextConfigurationTests.java b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementChildContextConfigurationTests.java new file mode 100644 index 000000000000..a46dca62ca63 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/java/org/springframework/boot/actuate/autoconfigure/web/servlet/ServletManagementChildContextConfigurationTests.java @@ -0,0 +1,49 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.autoconfigure.web.servlet; + +import org.junit.jupiter.api.Test; + +import org.springframework.boot.actuate.autoconfigure.web.servlet.ServletManagementChildContextConfiguration.AccessLogCustomizer; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link ServletManagementChildContextConfiguration}. + * + * @author Moritz Halbritter + */ +class ServletManagementChildContextConfigurationTests { + + @Test + void accessLogCustomizer() { + AccessLogCustomizer customizer = new AccessLogCustomizer("prefix") { + }; + assertThat(customizer.customizePrefix(null)).isEqualTo("prefix"); + assertThat(customizer.customizePrefix("existing")).isEqualTo("prefixexisting"); + assertThat(customizer.customizePrefix("prefixexisting")).isEqualTo("prefixexisting"); + } + + @Test + void accessLogCustomizerWithNullPrefix() { + AccessLogCustomizer customizer = new AccessLogCustomizer(null) { + }; + assertThat(customizer.customizePrefix(null)).isEqualTo(null); + assertThat(customizer.customizePrefix("existing")).isEqualTo("existing"); + } + +} diff --git a/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/resources/certificates/chains.p12 b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/resources/certificates/chains.p12 new file mode 100644 index 000000000000..b0a8d29a2b75 Binary files /dev/null and b/spring-boot-project/spring-boot-actuator-autoconfigure/src/test/resources/certificates/chains.p12 differ diff --git a/spring-boot-project/spring-boot-actuator/build.gradle b/spring-boot-project/spring-boot-actuator/build.gradle index 025dabe5021c..bd257c7cb469 100644 --- a/spring-boot-project/spring-boot-actuator/build.gradle +++ b/spring-boot-project/spring-boot-actuator/build.gradle @@ -39,9 +39,7 @@ dependencies { optional("io.micrometer:micrometer-registry-prometheus") optional("io.micrometer:micrometer-registry-prometheus-simpleclient") optional("io.prometheus:prometheus-metrics-exposition-formats") - optional("io.prometheus:simpleclient_pushgateway") { - exclude(group: "javax.xml.bind", module: "jaxb-api") - } + optional("io.prometheus:prometheus-metrics-exporter-pushgateway") optional("io.r2dbc:r2dbc-pool") optional("io.r2dbc:r2dbc-spi") optional("io.undertow:undertow-servlet") diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/amqp/RabbitHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/amqp/RabbitHealthIndicator.java index 386da31a9bc8..e4861b0bcc47 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/amqp/RabbitHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/amqp/RabbitHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ public class RabbitHealthIndicator extends AbstractHealthIndicator { public RabbitHealthIndicator(RabbitTemplate rabbitTemplate) { super("Rabbit health check failed"); - Assert.notNull(rabbitTemplate, "RabbitTemplate must not be null"); + Assert.notNull(rabbitTemplate, "'rabbitTemplate' must not be null"); this.rabbitTemplate = rabbitTemplate; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/AuditEvent.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/AuditEvent.java index e943f2d137a5..2fd00f99e25e 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/AuditEvent.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/AuditEvent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -83,8 +83,8 @@ public AuditEvent(String principal, String type, String... data) { * @param data the event data */ public AuditEvent(Instant timestamp, String principal, String type, Map data) { - Assert.notNull(timestamp, "Timestamp must not be null"); - Assert.notNull(type, "Type must not be null"); + Assert.notNull(timestamp, "'timestamp' must not be null"); + Assert.notNull(type, "'type' must not be null"); this.timestamp = timestamp; this.principal = (principal != null) ? principal : ""; this.type = type; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/AuditEventsEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/AuditEventsEndpoint.java index 57a2f1f0cfdd..cf001c1d44f9 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/AuditEventsEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/AuditEventsEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ public class AuditEventsEndpoint { private final AuditEventRepository auditEventRepository; public AuditEventsEndpoint(AuditEventRepository auditEventRepository) { - Assert.notNull(auditEventRepository, "AuditEventRepository must not be null"); + Assert.notNull(auditEventRepository, "'auditEventRepository' must not be null"); this.auditEventRepository = auditEventRepository; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/InMemoryAuditEventRepository.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/InMemoryAuditEventRepository.java index 06433c621e4b..668cd2074631 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/InMemoryAuditEventRepository.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/InMemoryAuditEventRepository.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -63,7 +63,7 @@ public void setCapacity(int capacity) { @Override public void add(AuditEvent event) { - Assert.notNull(event, "AuditEvent must not be null"); + Assert.notNull(event, "'event' must not be null"); synchronized (this.monitor) { this.tail = (this.tail + 1) % this.events.length; this.events[this.tail] = event; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/listener/AuditApplicationEvent.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/listener/AuditApplicationEvent.java index dd0e6f42f33e..f9fc139a80c7 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/listener/AuditApplicationEvent.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/audit/listener/AuditApplicationEvent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -77,7 +77,7 @@ public AuditApplicationEvent(Instant timestamp, String principal, String type, M */ public AuditApplicationEvent(AuditEvent auditEvent) { super(auditEvent); - Assert.notNull(auditEvent, "AuditEvent must not be null"); + Assert.notNull(auditEvent, "'auditEvent' must not be null"); this.auditEvent = auditEvent; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/availability/AvailabilityStateHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/availability/AvailabilityStateHealthIndicator.java index 5057f80110f2..34132169246f 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/availability/AvailabilityStateHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/availability/AvailabilityStateHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -55,9 +55,9 @@ public class AvailabilityStateHealthIndicator extends AbstractHealthIndicator { public AvailabilityStateHealthIndicator( ApplicationAvailability applicationAvailability, Class stateType, Consumer> statusMappings) { - Assert.notNull(applicationAvailability, "ApplicationAvailability must not be null"); - Assert.notNull(stateType, "StateType must not be null"); - Assert.notNull(statusMappings, "StatusMappings must not be null"); + Assert.notNull(applicationAvailability, "'applicationAvailability' must not be null"); + Assert.notNull(stateType, "'stateType' must not be null"); + Assert.notNull(statusMappings, "'statusMappings' must not be null"); this.applicationAvailability = applicationAvailability; this.stateType = stateType; statusMappings.accept(this.statusMappings::put); @@ -69,7 +69,7 @@ private void assertAllEnumsMapped(Class stateTy if (!this.statusMappings.containsKey(null) && Enum.class.isAssignableFrom(stateType)) { EnumSet elements = EnumSet.allOf((Class) stateType); for (Object element : elements) { - Assert.isTrue(this.statusMappings.containsKey(element), + Assert.state(this.statusMappings.containsKey(element), () -> "StatusMappings does not include " + element); } } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/cassandra/CassandraDriverHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/cassandra/CassandraDriverHealthIndicator.java index ed42a0bbda6a..c43e35f27393 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/cassandra/CassandraDriverHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/cassandra/CassandraDriverHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ public class CassandraDriverHealthIndicator extends AbstractHealthIndicator { */ public CassandraDriverHealthIndicator(CqlSession session) { super("Cassandra health check failed"); - Assert.notNull(session, "session must not be null"); + Assert.notNull(session, "'session' must not be null"); this.session = session; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/cassandra/CassandraDriverReactiveHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/cassandra/CassandraDriverReactiveHealthIndicator.java index 646560d0788c..eab8119ed9fc 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/cassandra/CassandraDriverReactiveHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/cassandra/CassandraDriverReactiveHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,7 +48,7 @@ public class CassandraDriverReactiveHealthIndicator extends AbstractReactiveHeal */ public CassandraDriverReactiveHealthIndicator(CqlSession session) { super("Cassandra health check failed"); - Assert.notNull(session, "session must not be null"); + Assert.notNull(session, "'session' must not be null"); this.session = session; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/couchbase/CouchbaseHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/couchbase/CouchbaseHealthIndicator.java index cefa92cc5424..3529270eb3a7 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/couchbase/CouchbaseHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/couchbase/CouchbaseHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,7 +42,7 @@ public class CouchbaseHealthIndicator extends AbstractHealthIndicator { */ public CouchbaseHealthIndicator(Cluster cluster) { super("Couchbase health check failed"); - Assert.notNull(cluster, "Cluster must not be null"); + Assert.notNull(cluster, "'cluster' must not be null"); this.cluster = cluster; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/elasticsearch/ElasticsearchReactiveHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/elasticsearch/ElasticsearchReactiveHealthIndicator.java index b3061d3494e4..789f2167bc2b 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/elasticsearch/ElasticsearchReactiveHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/elasticsearch/ElasticsearchReactiveHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -69,6 +69,7 @@ private Health processResponse(Health.Builder builder, HealthResponse response) builder.withDetail("task_max_waiting_in_queue_millis", response.taskMaxWaitingInQueueMillis()); builder.withDetail("active_shards_percent_as_number", Double.parseDouble(response.activeShardsPercentAsNumber())); + builder.withDetail("unassigned_primary_shards", response.unassignedPrimaryShards()); return builder.build(); } return builder.down().build(); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/mongo/MongoHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/mongo/MongoHealthIndicator.java index 307fa646c8c2..9e189ff98fdc 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/mongo/MongoHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/mongo/MongoHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ public class MongoHealthIndicator extends AbstractHealthIndicator { public MongoHealthIndicator(MongoTemplate mongoTemplate) { super("MongoDB health check failed"); - Assert.notNull(mongoTemplate, "MongoTemplate must not be null"); + Assert.notNull(mongoTemplate, "'mongoTemplate' must not be null"); this.mongoTemplate = mongoTemplate; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/mongo/MongoReactiveHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/mongo/MongoReactiveHealthIndicator.java index 39a3b6bb1428..9b5dd830400c 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/mongo/MongoReactiveHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/mongo/MongoReactiveHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ public class MongoReactiveHealthIndicator extends AbstractReactiveHealthIndicato public MongoReactiveHealthIndicator(ReactiveMongoTemplate reactiveMongoTemplate) { super("Mongo health check failed"); - Assert.notNull(reactiveMongoTemplate, "ReactiveMongoTemplate must not be null"); + Assert.notNull(reactiveMongoTemplate, "'reactiveMongoTemplate' must not be null"); this.reactiveMongoTemplate = reactiveMongoTemplate; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/redis/RedisHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/redis/RedisHealthIndicator.java index eaa3fbab86d9..69bd07271417 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/redis/RedisHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/data/redis/RedisHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,7 +40,7 @@ public class RedisHealthIndicator extends AbstractHealthIndicator { public RedisHealthIndicator(RedisConnectionFactory connectionFactory) { super("Redis health check failed"); - Assert.notNull(connectionFactory, "ConnectionFactory must not be null"); + Assert.notNull(connectionFactory, "'connectionFactory' must not be null"); this.redisConnectionFactory = connectionFactory; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/AbstractExposableEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/AbstractExposableEndpoint.java index 96108afa5388..ab4f2fdefc02 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/AbstractExposableEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/AbstractExposableEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,7 @@ public abstract class AbstractExposableEndpoint implements * @param id the endpoint id * @param enabledByDefault if the endpoint is enabled by default * @param operations the endpoint operations - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link #AbstractExposableEndpoint(EndpointId, Access, Collection)} */ @Deprecated(since = "3.4.0", forRemoval = true) @@ -57,8 +57,8 @@ public AbstractExposableEndpoint(EndpointId id, boolean enabledByDefault, Collec * @since 3.4.0 */ public AbstractExposableEndpoint(EndpointId id, Access defaultAccess, Collection operations) { - Assert.notNull(id, "ID must not be null"); - Assert.notNull(operations, "Operations must not be null"); + Assert.notNull(id, "'id' must not be null"); + Assert.notNull(operations, "'operations' must not be null"); this.id = id; this.defaultAccess = defaultAccess; this.operations = List.copyOf(operations); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/Access.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/Access.java index c810d540b20c..416f28d3c0ec 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/Access.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/Access.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ public enum Access { * @return this access if less than the maximum or the maximum permitted */ public Access cap(Access maxPermitted) { - Assert.notNull(maxPermitted, "'maxPermittedAccess' must not be null"); + Assert.notNull(maxPermitted, "'maxPermitted' must not be null"); return (ordinal() <= maxPermitted.ordinal()) ? this : maxPermitted; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/EndpointId.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/EndpointId.java index 269ca381aca3..a9e33ea65113 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/EndpointId.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/EndpointId.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -54,10 +54,10 @@ public final class EndpointId { private final String lowerCaseAlphaNumeric; private EndpointId(String value) { - Assert.hasText(value, "Value must not be empty"); - Assert.isTrue(VALID_PATTERN.matcher(value).matches(), "Value must only contain valid chars"); - Assert.isTrue(!Character.isDigit(value.charAt(0)), "Value must not start with a number"); - Assert.isTrue(!Character.isUpperCase(value.charAt(0)), "Value must not start with an uppercase letter"); + Assert.hasText(value, "'value' must not be empty"); + Assert.isTrue(VALID_PATTERN.matcher(value).matches(), "'value' must only contain valid chars"); + Assert.isTrue(!Character.isDigit(value.charAt(0)), "'value' must not start with a number"); + Assert.isTrue(!Character.isUpperCase(value.charAt(0)), "'value' must not start with an uppercase letter"); if (WARNING_PATTERN.matcher(value).find()) { logWarning(value); } @@ -125,7 +125,7 @@ public static EndpointId of(String value) { * @since 2.2.0 */ public static EndpointId of(Environment environment, String value) { - Assert.notNull(environment, "Environment must not be null"); + Assert.notNull(environment, "'environment' must not be null"); return new EndpointId(migrateLegacyId(environment, value)); } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/ExposableEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/ExposableEndpoint.java index 7de3a6ab7d57..34f8dc6d956b 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/ExposableEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/ExposableEndpoint.java @@ -37,7 +37,7 @@ public interface ExposableEndpoint { /** * Returns if the endpoint is enabled by default. * @return if the endpoint is enabled by default - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link #getDefaultAccess()} */ @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/InvocationContext.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/InvocationContext.java index 3369570c391e..e1853cf9c8d7 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/InvocationContext.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/InvocationContext.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,8 +48,8 @@ public class InvocationContext { */ public InvocationContext(SecurityContext securityContext, Map arguments, OperationArgumentResolver... argumentResolvers) { - Assert.notNull(securityContext, "SecurityContext must not be null"); - Assert.notNull(arguments, "Arguments must not be null"); + Assert.notNull(securityContext, "'securityContext' must not be null"); + Assert.notNull(arguments, "'arguments' must not be null"); this.arguments = arguments; this.argumentResolvers = new ArrayList<>(); if (argumentResolvers != null) { diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/OperationArgumentResolver.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/OperationArgumentResolver.java index 9ade2ec523d9..539f92fa5e08 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/OperationArgumentResolver.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/OperationArgumentResolver.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,8 +53,8 @@ public interface OperationArgumentResolver { * @return an {@link OperationArgumentResolver} instance */ static OperationArgumentResolver of(Class type, Supplier supplier) { - Assert.notNull(type, "Type must not be null"); - Assert.notNull(supplier, "Supplier must not be null"); + Assert.notNull(type, "'type' must not be null"); + Assert.notNull(supplier, "'supplier' must not be null"); return new OperationArgumentResolver() { @Override diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/SanitizableData.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/SanitizableData.java index eaaa15c54d03..3e47035ada55 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/SanitizableData.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/SanitizableData.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,6 +16,8 @@ package org.springframework.boot.actuate.endpoint; +import java.util.Locale; + import org.springframework.core.env.PropertySource; /** @@ -36,6 +38,8 @@ public final class SanitizableData { private final String key; + private String lowerCaseKey; + private final Object value; /** @@ -67,6 +71,20 @@ public String getKey() { return this.key; } + /** + * Return the key as a lowercase value. + * @return the key as a lowercase value + * @since 3.5.0 + */ + public String getLowerCaseKey() { + String result = this.lowerCaseKey; + if (result == null && this.key != null) { + result = this.key.toLowerCase(Locale.getDefault()); + this.lowerCaseKey = result; + } + return result; + } + /** * Return the value of the data. * @return the data value diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/Sanitizer.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/Sanitizer.java index 3ad0497f790d..4c58388e297b 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/Sanitizer.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/Sanitizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -72,7 +72,7 @@ public Object sanitize(SanitizableData data, boolean showUnsanitized) { return SanitizableData.SANITIZED_VALUE; } for (SanitizingFunction sanitizingFunction : this.sanitizingFunctions) { - data = sanitizingFunction.apply(data); + data = sanitizingFunction.applyUnlessFiltered(data); Object sanitizedValue = data.getValue(); if (!value.equals(sanitizedValue)) { return sanitizedValue; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/SanitizingFunction.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/SanitizingFunction.java index 3d9ffecefea0..ccbed309c224 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/SanitizingFunction.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/SanitizingFunction.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,12 +16,27 @@ package org.springframework.boot.actuate.endpoint; +import java.util.Arrays; +import java.util.List; +import java.util.Locale; +import java.util.function.BiPredicate; +import java.util.function.Predicate; +import java.util.regex.Pattern; + +import org.springframework.util.Assert; + /** * Function that takes a {@link SanitizableData} and applies sanitization to the value, if * necessary. Can be used by a {@link Sanitizer} to determine the sanitized value. + *

+ * This interface also provides convenience methods that can help build a + * {@link SanitizingFunction} instances, for example to return from a {@code @Bean} + * method. See {@link #sanitizeValue()} for an example. * * @author Madhura Bhave + * @author Phillip Webb * @since 2.6.0 + * @see Sanitizer */ @FunctionalInterface public interface SanitizingFunction { @@ -33,4 +48,401 @@ public interface SanitizingFunction { */ SanitizableData apply(SanitizableData data); + /** + * Return an optional filter that determines if the sanitizing function applies. + * @return a predicate used to filter functions or {@code null} if no filter is + * declared + * @since 3.5.0 + * @see #applyUnlessFiltered(SanitizableData) + */ + default Predicate filter() { + return null; + } + + /** + * Apply the sanitizing function as long as the filter passes or there is no filter. + * @param data the data to sanitize + * @return the sanitized data or the original instance is no sanitization is applied + * @since 3.5.0 + */ + default SanitizableData applyUnlessFiltered(SanitizableData data) { + return (filter() == null || filter().test(data)) ? apply(data) : data; + } + + /** + * Return a new function with a filter that also applies if the data is + * likely to contain a sensitive value. This method can help construct a useful + * sanitizing function, but may not catch all sensitive data so care should be taken + * to test the results for your specific environment. + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifLikelySensitive() { + return ifLikelyCredential().ifLikelyUri().ifLikelySensitiveProperty().ifVcapServices(); + } + + /** + * Return a new function with a filter that also applies if the data is + * likely to contain a credential. This method can help construct a useful sanitizing + * function, but may not catch all sensitive data so care should be taken to test the + * results for your specific environment. + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifLikelyCredential() { + return ifKeyEndsWith("password", "secret", "key", "token").ifKeyContains("credentials"); + } + + /** + * Return a new function with a filter that also applies if the data is + * likely to contain a URI. This method can help construct a useful sanitizing + * function, but may not catch all sensitive data so care should be taken to test the + * results for your specific environment. + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifLikelyUri() { + return ifKeyEndsWith("uri", "uris", "url", "urls", "address", "addresses"); + } + + /** + * Return a new function with a filter that also applies if the data is + * likely to contain a sensitive property value. This method can help construct a + * useful sanitizing function, but may not catch all sensitive data so care should be + * taken to test the results for your specific environment. + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifLikelySensitiveProperty() { + return ifKeyMatches("sun.java.command", "^spring[._]application[._]json$"); + } + + /** + * Return a new function with a filter that also applies if the data is for + * VCAP services. + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + + default SanitizingFunction ifVcapServices() { + return ifKeyEquals("vcap_services").ifKeyMatches("^vcap\\.services.*$"); + } + + /** + * Return a new function with a filter that also applies if the data key is + * equal to any of the given values (ignoring case). + * @param values the case insensitive values that the key can equal + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifKeyEquals(String... values) { + Assert.notNull(values, "'values' must not be null"); + return ifKeyMatchesIgnoringCase(String::equals, values); + } + + /** + * Return a new function with a filter that also applies if the data key ends + * with any of the given values (ignoring case). + * @param suffixes the case insensitive suffixes that they key can end with + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifKeyEndsWith(String... suffixes) { + Assert.notNull(suffixes, "'suffixes' must not be null"); + return ifKeyMatchesIgnoringCase(String::endsWith, suffixes); + } + + /** + * Return a new function with a filter that also applies if the data key + * contains any of the given values (ignoring case). + * @param values the case insensitive values that the key can contain + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifKeyContains(String... values) { + Assert.notNull(values, "'values' must not be null"); + return ifKeyMatchesIgnoringCase(String::contains, values); + } + + /** + * Return a new function with a filter that also applies if the data key and + * any of the values match the given predicate. The predicate is only called with + * lower case values. + * @param predicate the predicate used to check the key against a value. The key is + * the first argument and the value is the second. Both are converted to lower case + * @param values the case insensitive values that the key can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifKeyMatchesIgnoringCase(BiPredicate predicate, String... values) { + Assert.notNull(predicate, "'predicate' must not be null"); + Assert.notNull(values, "'values' must not be null"); + return ifMatches(Arrays.stream(values).map((value) -> onKeyIgnoringCase(predicate, value)).toList()); + } + + /** + * Return a new function with a filter that also applies if the data key + * matches any of the given regex patterns (ignoring case). + * @param regexes the case insensitive regexes that the key can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifKeyMatches(String... regexes) { + Assert.notNull(regexes, "'regexes' must not be null"); + return ifKeyMatches(Arrays.stream(regexes).map(this::caseInsensitivePattern).toArray(Pattern[]::new)); + } + + /** + * Return a new function with a filter that also applies if the data key + * matches any of the given patterns. + * @param patterns the patterns that the key can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifKeyMatches(Pattern... patterns) { + Assert.notNull(patterns, "'patterns' must not be null"); + return ifKeyMatches(Arrays.stream(patterns).map(Pattern::asMatchPredicate).toList()); + } + + /** + * Return a new function with a filter that also applies if the data key + * matches any of the given predicates. + * @param predicates the predicates that the key can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifKeyMatches(List> predicates) { + Assert.notNull(predicates, "'predicates' must not be null"); + return ifMatches(predicates.stream().map(this::onKey).toList()); + } + + /** + * Return a new function with a filter that also applies if the data key + * matches any of the given predicate. + * @param predicate the predicate that the key can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifKeyMatches(Predicate predicate) { + Assert.notNull(predicate, "'predicate' must not be null"); + return ifMatches(onKey(predicate)); + } + + /** + * Return a new function with a filter that also applies if the data string + * value matches any of the given regex patterns (ignoring case). + * @param regexes the case insensitive regexes that the values string can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifValueStringMatches(String... regexes) { + Assert.notNull(regexes, "'regexes' must not be null"); + return ifValueStringMatches(Arrays.stream(regexes).map(this::caseInsensitivePattern).toArray(Pattern[]::new)); + } + + /** + * Return a new function with a filter that also applies if the data string + * value matches any of the given patterns. + * @param patterns the patterns that the value string can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifValueStringMatches(Pattern... patterns) { + Assert.notNull(patterns, "'patterns' must not be null"); + return ifValueStringMatches(Arrays.stream(patterns).map(Pattern::asMatchPredicate).toList()); + } + + /** + * Return a new function with a filter that also applies if the data string + * value matches any of the given predicates. + * @param predicates the predicates that the value string can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + + default SanitizingFunction ifValueStringMatches(List> predicates) { + Assert.notNull(predicates, "'predicates' must not be null"); + return ifMatches(predicates.stream().map(this::onValueString).toList()); + } + + /** + * Return a new function with a filter that also applies if the data value + * matches any of the given predicates. + * @param predicates the predicates that the value can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifValueMatches(List> predicates) { + Assert.notNull(predicates, "'predicates' must not be null"); + return ifMatches(predicates.stream().map(this::onValue).toList()); + } + + /** + * Return a new function with a filter that also applies if the data string + * value matches the given predicate. + * @param predicate the predicate that the value string can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + + default SanitizingFunction ifValueStringMatches(Predicate predicate) { + Assert.notNull(predicate, "'predicate' must not be null"); + return ifMatches(onValueString(predicate)); + } + + /** + * Return a new function with a filter that also applies if the data value + * matches the given predicate. + * @param predicate the predicate that the value can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifValueMatches(Predicate predicate) { + Assert.notNull(predicate, "'predicate' must not be null"); + return ifMatches((data) -> predicate.test(data.getValue())); + } + + /** + * Return a new function with a filter that also applies if the data matches + * any of the given predicates. + * @param predicates the predicates that the data can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifMatches(List> predicates) { + Assert.notNull(predicates, "'predicates' must not be null"); + Predicate combined = null; + for (Predicate predicate : predicates) { + combined = (combined != null) ? combined.or(predicate) : predicate; + } + return ifMatches(combined); + } + + /** + * Return a new function with a filter that also applies if the data matches + * the given predicate. + * @param predicate the predicate that the data can match + * @return a new sanitizing function with an updated {@link #filter()} + * @since 3.5.0 + * @see #filter() + * @see #sanitizeValue() + */ + default SanitizingFunction ifMatches(Predicate predicate) { + Assert.notNull(predicate, "'predicate' must not be null"); + Predicate filter = (filter() != null) ? filter().or(predicate) : predicate; + return new SanitizingFunction() { + + @Override + public Predicate filter() { + return filter; + } + + @Override + public SanitizableData apply(SanitizableData data) { + return SanitizingFunction.this.apply(data); + } + + }; + } + + private Pattern caseInsensitivePattern(String regex) { + Assert.notNull(regex, "'regex' must not be null"); + return Pattern.compile(regex, Pattern.CASE_INSENSITIVE); + } + + private Predicate onKeyIgnoringCase(BiPredicate predicate, String value) { + Assert.notNull(predicate, "'predicate' must not be null"); + Assert.notNull(value, "'value' must not be null"); + String lowerCaseValue = value.toLowerCase(Locale.getDefault()); + return (data) -> nullSafeTest(data.getLowerCaseKey(), + (lowerCaseKey) -> predicate.test(lowerCaseKey, lowerCaseValue)); + } + + private Predicate onKey(Predicate predicate) { + Assert.notNull(predicate, "'predicate' must not be null"); + return (data) -> nullSafeTest(data.getKey(), predicate); + } + + private Predicate onValue(Predicate predicate) { + Assert.notNull(predicate, "'predicate' must not be null"); + return (data) -> nullSafeTest(data.getValue(), predicate); + } + + private Predicate onValueString(Predicate predicate) { + Assert.notNull(predicate, "'predicate' must not be null"); + return (data) -> nullSafeTest((data.getValue() != null) ? data.getValue().toString() : null, predicate); + } + + private boolean nullSafeTest(T value, Predicate predicate) { + return value != null && predicate.test(value); + } + + /** + * Factory method to return a {@link SanitizingFunction} that sanitizes the value. + * This method is often chained with one or more {@code if...} methods. For example: + *
+	 * return SanitizingFunction.sanitizeValue()
+	 * 	.ifKeyContains("password", "secret")
+	 * 	.ifValueStringMatches("^gh._[a-zA-Z0-9]{36}$");
+	 * 
+ * @return a {@link SanitizingFunction} that sanitizes values. + */ + static SanitizingFunction sanitizeValue() { + return SanitizableData::withSanitizedValue; + } + + /** + * Helper method that can be used working with a sanitizingFunction as a lambda. For + * example:
+	 * SanitizingFunction.of((data) -> data.withValue("----")).ifKeyContains("password");
+	 * 
+ * @param sanitizingFunction the sanitizing function lambda + * @return a {@link SanitizingFunction} for further method calls + * @since 3.5.0 + */ + static SanitizingFunction of(SanitizingFunction sanitizingFunction) { + Assert.notNull(sanitizingFunction, "'sanitizingFunction' must not be null"); + return sanitizingFunction; + } + } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/AbstractDiscoveredEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/AbstractDiscoveredEndpoint.java index 72a581810331..95512c1b2612 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/AbstractDiscoveredEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/AbstractDiscoveredEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,7 +48,7 @@ public abstract class AbstractDiscoveredEndpoint extends Ab * @param id the ID of the endpoint * @param enabledByDefault if the endpoint is enabled by default * @param operations the endpoint operations - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link #AbstractDiscoveredEndpoint(EndpointDiscoverer, Object, EndpointId, Access, Collection)} */ @SuppressWarnings("removal") @@ -70,8 +70,8 @@ public AbstractDiscoveredEndpoint(EndpointDiscoverer discoverer, Object en public AbstractDiscoveredEndpoint(EndpointDiscoverer discoverer, Object endpointBean, EndpointId id, Access defaultAccess, Collection operations) { super(id, defaultAccess, operations); - Assert.notNull(discoverer, "Discoverer must not be null"); - Assert.notNull(endpointBean, "EndpointBean must not be null"); + Assert.notNull(discoverer, "'discoverer' must not be null"); + Assert.notNull(endpointBean, "'endpointBean' must not be null"); this.discoverer = discoverer; this.endpointBean = endpointBean; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/DiscoveredOperationMethod.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/DiscoveredOperationMethod.java index 4cf89333a905..e72059365c14 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/DiscoveredOperationMethod.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/DiscoveredOperationMethod.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,7 @@ public class DiscoveredOperationMethod extends OperationMethod { public DiscoveredOperationMethod(Method method, OperationType operationType, AnnotationAttributes annotationAttributes) { super(method, operationType); - Assert.notNull(annotationAttributes, "AnnotationAttributes must not be null"); + Assert.notNull(annotationAttributes, "'annotationAttributes' must not be null"); List producesMediaTypes = new ArrayList<>(); producesMediaTypes.addAll(Arrays.asList(annotationAttributes.getStringArray("produces"))); producesMediaTypes.addAll(getProducesFromProducible(annotationAttributes)); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/DiscovererEndpointFilter.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/DiscovererEndpointFilter.java index 45c0354b880a..154051a8149f 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/DiscovererEndpointFilter.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/DiscovererEndpointFilter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ public abstract class DiscovererEndpointFilter implements EndpointFilter> discoverer) { - Assert.notNull(discoverer, "Discoverer must not be null"); + Assert.notNull(discoverer, "'discoverer' must not be null"); this.discoverer = discoverer; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/Endpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/Endpoint.java index 496bf12e801e..d52fee56bfbf 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/Endpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/Endpoint.java @@ -66,7 +66,7 @@ /** * If the endpoint should be enabled or disabled by default. * @return {@code true} if the endpoint is enabled by default - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of {@link #defaultAccess()} + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of {@link #defaultAccess()} */ @Deprecated(since = "3.4.0", forRemoval = true) boolean enableByDefault() default true; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/EndpointDiscoverer.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/EndpointDiscoverer.java index d28113bbb268..09b12e99d9af 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/EndpointDiscoverer.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/annotation/EndpointDiscoverer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -90,7 +90,7 @@ public abstract class EndpointDiscoverer, O exten * @param parameterValueMapper the parameter value mapper * @param invokerAdvisors invoker advisors to apply * @param endpointFilters endpoint filters to apply - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link #EndpointDiscoverer(ApplicationContext, ParameterValueMapper, Collection, Collection, Collection)} */ @Deprecated(since = "3.4.0", forRemoval = true) @@ -111,11 +111,11 @@ public EndpointDiscoverer(ApplicationContext applicationContext, ParameterValueM public EndpointDiscoverer(ApplicationContext applicationContext, ParameterValueMapper parameterValueMapper, Collection invokerAdvisors, Collection> endpointFilters, Collection> operationFilters) { - Assert.notNull(applicationContext, "ApplicationContext must not be null"); - Assert.notNull(parameterValueMapper, "ParameterValueMapper must not be null"); - Assert.notNull(invokerAdvisors, "InvokerAdvisors must not be null"); - Assert.notNull(endpointFilters, "EndpointFilters must not be null"); - Assert.notNull(operationFilters, "OperationFilters must not be null"); + Assert.notNull(applicationContext, "'applicationContext' must not be null"); + Assert.notNull(parameterValueMapper, "'parameterValueMapper' must not be null"); + Assert.notNull(invokerAdvisors, "'invokerAdvisors' must not be null"); + Assert.notNull(endpointFilters, "'endpointFilters' must not be null"); + Assert.notNull(operationFilters, "'operationFilters' must not be null"); this.applicationContext = applicationContext; this.endpointFilters = Collections.unmodifiableCollection(endpointFilters); this.operationFilters = Collections.unmodifiableCollection(operationFilters); @@ -388,7 +388,7 @@ protected Class getEndpointType() { * @param enabledByDefault if the endpoint is enabled by default * @param operations the endpoint operations * @return a created endpoint (a {@link DiscoveredEndpoint} is recommended) - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link #createEndpoint(Object, EndpointId, Access, Collection)} */ @Deprecated(since = "3.4.0", forRemoval = true) @@ -440,8 +440,8 @@ protected static final class OperationKey { * @param description a human-readable description of the key */ public OperationKey(Object key, Supplier description) { - Assert.notNull(key, "Key must not be null"); - Assert.notNull(description, "Description must not be null"); + Assert.notNull(key, "'key' must not be null"); + Assert.notNull(description, "'description' must not be null"); this.key = key; this.description = description; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/convert/ConversionServiceParameterValueMapper.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/convert/ConversionServiceParameterValueMapper.java index 3d185879ed51..666b00ea2c22 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/convert/ConversionServiceParameterValueMapper.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/convert/ConversionServiceParameterValueMapper.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ public ConversionServiceParameterValueMapper() { * @param conversionService the conversion service */ public ConversionServiceParameterValueMapper(ConversionService conversionService) { - Assert.notNull(conversionService, "ConversionService must not be null"); + Assert.notNull(conversionService, "'conversionService' must not be null"); this.conversionService = conversionService; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethod.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethod.java index 242de67e743c..441b4334c991 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethod.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethod.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,8 +48,8 @@ public class OperationMethod { * @param operationType the operation type */ public OperationMethod(Method method, OperationType operationType) { - Assert.notNull(method, "Method must not be null"); - Assert.notNull(operationType, "OperationType must not be null"); + Assert.notNull(method, "'method' must not be null"); + Assert.notNull(operationType, "'operationType' must not be null"); this.method = method; this.operationType = operationType; this.operationParameters = new OperationMethodParameters(method, DEFAULT_PARAMETER_NAME_DISCOVERER); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodParameters.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodParameters.java index d81826a8603d..171b8ffb5164 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodParameters.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodParameters.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -44,8 +44,8 @@ class OperationMethodParameters implements OperationParameters { * @param parameterNameDiscoverer the parameter name discoverer */ OperationMethodParameters(Method method, ParameterNameDiscoverer parameterNameDiscoverer) { - Assert.notNull(method, "Method must not be null"); - Assert.notNull(parameterNameDiscoverer, "ParameterNameDiscoverer must not be null"); + Assert.notNull(method, "'method' must not be null"); + Assert.notNull(parameterNameDiscoverer, "'parameterNameDiscoverer' must not be null"); String[] parameterNames = parameterNameDiscoverer.getParameterNames(method); Parameter[] parameters = method.getParameters(); Assert.state(parameterNames != null, () -> "Failed to extract parameter names for " + method); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/ReflectiveOperationInvoker.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/ReflectiveOperationInvoker.java index 7f8f3960b4f1..505a2e2f8527 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/ReflectiveOperationInvoker.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoke/reflect/ReflectiveOperationInvoker.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,9 +56,9 @@ public class ReflectiveOperationInvoker implements OperationInvoker { */ public ReflectiveOperationInvoker(Object target, OperationMethod operationMethod, ParameterValueMapper parameterValueMapper) { - Assert.notNull(target, "Target must not be null"); - Assert.notNull(operationMethod, "OperationMethod must not be null"); - Assert.notNull(parameterValueMapper, "ParameterValueMapper must not be null"); + Assert.notNull(target, "'target' must not be null"); + Assert.notNull(operationMethod, "'operationMethod' must not be null"); + Assert.notNull(parameterValueMapper, "'parameterValueMapper' must not be null"); ReflectionUtils.makeAccessible(operationMethod.getMethod()); this.target = target; this.operationMethod = operationMethod; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoker/cache/CachingOperationInvoker.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoker/cache/CachingOperationInvoker.java index 8cd51645de92..df24c8e0c34b 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoker/cache/CachingOperationInvoker.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/invoker/cache/CachingOperationInvoker.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -65,7 +65,7 @@ public class CachingOperationInvoker implements OperationInvoker { * @param timeToLive the maximum time in milliseconds that a response can be cached */ CachingOperationInvoker(OperationInvoker invoker, long timeToLive) { - Assert.isTrue(timeToLive > 0, "TimeToLive must be strictly positive"); + Assert.isTrue(timeToLive > 0, "'timeToLive' must be greater than zero"); this.invoker = invoker; this.timeToLive = timeToLive; this.cachedResponses = new ConcurrentReferenceHashMap<>(); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/EndpointMBean.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/EndpointMBean.java index 845e9707c7e5..1bb56a0ddb28 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/EndpointMBean.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/EndpointMBean.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -63,8 +63,8 @@ public class EndpointMBean implements DynamicMBean { private final Map operations; EndpointMBean(JmxOperationResponseMapper responseMapper, ClassLoader classLoader, ExposableJmxEndpoint endpoint) { - Assert.notNull(responseMapper, "ResponseMapper must not be null"); - Assert.notNull(endpoint, "Endpoint must not be null"); + Assert.notNull(responseMapper, "'responseMapper' must not be null"); + Assert.notNull(endpoint, "'endpoint' must not be null"); this.responseMapper = responseMapper; this.classLoader = classLoader; this.endpoint = endpoint; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/JmxEndpointExporter.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/JmxEndpointExporter.java index 7c7f147db521..159e010d22f5 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/JmxEndpointExporter.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/JmxEndpointExporter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -61,10 +61,10 @@ public class JmxEndpointExporter implements InitializingBean, DisposableBean, Be public JmxEndpointExporter(MBeanServer mBeanServer, EndpointObjectNameFactory objectNameFactory, JmxOperationResponseMapper responseMapper, Collection endpoints) { - Assert.notNull(mBeanServer, "MBeanServer must not be null"); - Assert.notNull(objectNameFactory, "ObjectNameFactory must not be null"); - Assert.notNull(responseMapper, "ResponseMapper must not be null"); - Assert.notNull(endpoints, "Endpoints must not be null"); + Assert.notNull(mBeanServer, "'mBeanServer' must not be null"); + Assert.notNull(objectNameFactory, "'objectNameFactory' must not be null"); + Assert.notNull(responseMapper, "'responseMapper' must not be null"); + Assert.notNull(endpoints, "'endpoints' must not be null"); this.mBeanServer = mBeanServer; this.objectNameFactory = objectNameFactory; this.responseMapper = responseMapper; @@ -95,7 +95,7 @@ private boolean hasOperations(ExposableJmxEndpoint endpoint) { } private ObjectName register(ExposableJmxEndpoint endpoint) { - Assert.notNull(endpoint, "Endpoint must not be null"); + Assert.notNull(endpoint, "'endpoint' must not be null"); try { ObjectName name = this.objectNameFactory.getObjectName(endpoint); EndpointMBean mbean = new EndpointMBean(this.responseMapper, this.classLoader, endpoint); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/annotation/JmxEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/annotation/JmxEndpoint.java index ec6761464fd0..02916b5cdb6d 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/annotation/JmxEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/annotation/JmxEndpoint.java @@ -51,7 +51,7 @@ /** * If the endpoint should be enabled or disabled by default. * @return {@code true} if the endpoint is enabled by default - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of */ @Deprecated(since = "3.4.0", forRemoval = true) @AliasFor(annotation = Endpoint.class) diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/annotation/JmxEndpointDiscoverer.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/annotation/JmxEndpointDiscoverer.java index 8a2f1bf2b756..d9f235def88b 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/annotation/JmxEndpointDiscoverer.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/jmx/annotation/JmxEndpointDiscoverer.java @@ -54,7 +54,7 @@ public class JmxEndpointDiscoverer extends EndpointDiscoverer producedAndConsumed) { * @param consumed the default media types that are consumed by an endpoint. Must not */ public EndpointMediaTypes(List produced, List consumed) { - Assert.notNull(produced, "Produced must not be null"); - Assert.notNull(consumed, "Consumed must not be null"); + Assert.notNull(produced, "'produced' must not be null"); + Assert.notNull(consumed, "'consumed' must not be null"); this.produced = Collections.unmodifiableList(produced); this.consumed = Collections.unmodifiableList(consumed); } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/EndpointServlet.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/EndpointServlet.java index 6f7b9e83ae16..a419c1e827c7 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/EndpointServlet.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/EndpointServlet.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,7 @@ public EndpointServlet(Class servlet) { } private static Servlet instantiateClass(Class servlet) { - Assert.notNull(servlet, "Servlet must not be null"); + Assert.notNull(servlet, "'servlet' must not be null"); return BeanUtils.instantiateClass(servlet); } @@ -58,21 +58,21 @@ public EndpointServlet(Servlet servlet) { } private EndpointServlet(Servlet servlet, Map initParameters, int loadOnStartup) { - Assert.notNull(servlet, "Servlet must not be null"); + Assert.notNull(servlet, "'servlet' must not be null"); this.servlet = servlet; this.initParameters = Collections.unmodifiableMap(initParameters); this.loadOnStartup = loadOnStartup; } public EndpointServlet withInitParameter(String name, String value) { - Assert.hasText(name, "Name must not be empty"); + Assert.hasText(name, "'name' must not be empty"); return withInitParameters(Collections.singletonMap(name, value)); } public EndpointServlet withInitParameters(Map initParameters) { - Assert.notNull(initParameters, "InitParameters must not be null"); + Assert.notNull(initParameters, "'initParameters' must not be null"); boolean hasEmptyName = initParameters.keySet().stream().anyMatch((name) -> !StringUtils.hasText(name)); - Assert.isTrue(!hasEmptyName, "InitParameters must not contain empty names"); + Assert.isTrue(!hasEmptyName, "'initParameters' must not contain empty names"); Map mergedInitParameters = new LinkedHashMap<>(this.initParameters); mergedInitParameters.putAll(initParameters); return new EndpointServlet(this.servlet, mergedInitParameters, this.loadOnStartup); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/Link.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/Link.java index f0e1b9bcb366..86b84dec0dd4 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/Link.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/Link.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ public class Link { * @param href the href */ public Link(String href) { - Assert.notNull(href, "HREF must not be null"); + Assert.notNull(href, "'href' must not be null"); this.href = href; this.templated = href.contains("{"); } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/PathMappedEndpoints.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/PathMappedEndpoints.java index 96995b2e0daf..aa1946124ac2 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/PathMappedEndpoints.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/PathMappedEndpoints.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ public class PathMappedEndpoints implements Iterable { * @param supplier the endpoint supplier */ public PathMappedEndpoints(String basePath, EndpointsSupplier supplier) { - Assert.notNull(supplier, "Supplier must not be null"); + Assert.notNull(supplier, "'supplier' must not be null"); this.basePath = (basePath != null) ? basePath : ""; this.endpoints = getEndpoints(Collections.singleton(supplier)); } @@ -58,7 +58,7 @@ public PathMappedEndpoints(String basePath, EndpointsSupplier supplier) { * @param suppliers the endpoint suppliers */ public PathMappedEndpoints(String basePath, Collection> suppliers) { - Assert.notNull(suppliers, "Suppliers must not be null"); + Assert.notNull(suppliers, "'suppliers' must not be null"); this.basePath = (basePath != null) ? basePath : ""; this.endpoints = getEndpoints(suppliers); } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/PathMapper.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/PathMapper.java index ba59d987cf3f..52639110ca20 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/PathMapper.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/PathMapper.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,7 @@ public interface PathMapper { * @return the path of the endpoint */ static String getRootPath(List pathMappers, EndpointId endpointId) { - Assert.notNull(endpointId, "EndpointId must not be null"); + Assert.notNull(endpointId, "'endpointId' must not be null"); if (pathMappers != null) { for (PathMapper mapper : pathMappers) { String path = mapper.getRootPath(endpointId); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/ServletEndpointRegistrar.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/ServletEndpointRegistrar.java index 13508e7c87b8..3c82519d1c39 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/ServletEndpointRegistrar.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/ServletEndpointRegistrar.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -70,7 +70,7 @@ public ServletEndpointRegistrar(String basePath, Collection servletEndpoints, EndpointAccessResolver endpointAccessResolver) { - Assert.notNull(servletEndpoints, "ServletEndpoints must not be null"); + Assert.notNull(servletEndpoints, "'servletEndpoints' must not be null"); this.basePath = cleanBasePath(basePath); this.servletEndpoints = servletEndpoints; this.endpointAccessResolver = endpointAccessResolver; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/RequestPredicateFactory.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/RequestPredicateFactory.java index 1f9557bc8e0a..adcd4d52844e 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/RequestPredicateFactory.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/RequestPredicateFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ class RequestPredicateFactory { private final EndpointMediaTypes endpointMediaTypes; RequestPredicateFactory(EndpointMediaTypes endpointMediaTypes) { - Assert.notNull(endpointMediaTypes, "EndpointMediaTypes must not be null"); + Assert.notNull(endpointMediaTypes, "'endpointMediaTypes' must not be null"); this.endpointMediaTypes = endpointMediaTypes; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/WebEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/WebEndpoint.java index adca3f24d477..eb0f48ca6e31 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/WebEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/WebEndpoint.java @@ -51,7 +51,7 @@ /** * If the endpoint should be enabled or disabled by default. * @return {@code true} if the endpoint is enabled by default - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of {@link #defaultAccess()} */ @Deprecated(since = "3.4.0", forRemoval = true) @AliasFor(annotation = Endpoint.class) diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/WebEndpointDiscoverer.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/WebEndpointDiscoverer.java index 9d398212f152..d984a3854f52 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/WebEndpointDiscoverer.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/annotation/WebEndpointDiscoverer.java @@ -67,7 +67,7 @@ public class WebEndpointDiscoverer extends EndpointDiscoverer endpoints, CorsConfiguration corsConfiguration, EndpointAccessResolver endpointAccessResolver) { - Assert.notNull(endpointMapping, "EndpointMapping must not be null"); - Assert.notNull(endpoints, "Endpoints must not be null"); + Assert.notNull(endpointMapping, "'endpointMapping' must not be null"); + Assert.notNull(endpoints, "'endpoints' must not be null"); this.endpointMapping = endpointMapping; this.handlers = getHandlers(endpoints); this.corsConfiguration = corsConfiguration; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/servlet/ControllerEndpointHandlerMapping.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/servlet/ControllerEndpointHandlerMapping.java index b7085b91717e..267614cdf7d1 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/servlet/ControllerEndpointHandlerMapping.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/endpoint/web/servlet/ControllerEndpointHandlerMapping.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -88,8 +88,8 @@ public ControllerEndpointHandlerMapping(EndpointMapping endpointMapping, public ControllerEndpointHandlerMapping(EndpointMapping endpointMapping, Collection endpoints, CorsConfiguration corsConfiguration, EndpointAccessResolver endpointAccessResolver) { - Assert.notNull(endpointMapping, "EndpointMapping must not be null"); - Assert.notNull(endpoints, "Endpoints must not be null"); + Assert.notNull(endpointMapping, "'endpointMapping' must not be null"); + Assert.notNull(endpoints, "'endpoints' must not be null"); this.endpointMapping = endpointMapping; this.handlers = getHandlers(endpoints); this.corsConfiguration = corsConfiguration; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/hazelcast/HazelcastHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/hazelcast/HazelcastHealthIndicator.java index d28b82a8e7d6..694ebec52a97 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/hazelcast/HazelcastHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/hazelcast/HazelcastHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,7 +36,7 @@ public class HazelcastHealthIndicator extends AbstractHealthIndicator { public HazelcastHealthIndicator(HazelcastInstance hazelcast) { super("Hazelcast health check failed"); - Assert.notNull(hazelcast, "HazelcastInstance must not be null"); + Assert.notNull(hazelcast, "'hazelcast' must not be null"); this.hazelcast = hazelcast; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AbstractHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AbstractHealthIndicator.java index ebf13309c9e0..a89f26f8b2ac 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AbstractHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AbstractHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -71,7 +71,7 @@ protected AbstractHealthIndicator(String healthCheckFailedMessage) { * @since 2.0.0 */ protected AbstractHealthIndicator(Function healthCheckFailedMessage) { - Assert.notNull(healthCheckFailedMessage, "HealthCheckFailedMessage must not be null"); + Assert.notNull(healthCheckFailedMessage, "'healthCheckFailedMessage' must not be null"); this.healthCheckFailedMessage = healthCheckFailedMessage; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AbstractReactiveHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AbstractReactiveHealthIndicator.java index 0b6e2c792693..81ffd0a3234d 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AbstractReactiveHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AbstractReactiveHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -70,7 +70,7 @@ protected AbstractReactiveHealthIndicator(String healthCheckFailedMessage) { * @since 2.1.7 */ protected AbstractReactiveHealthIndicator(Function healthCheckFailedMessage) { - Assert.notNull(healthCheckFailedMessage, "HealthCheckFailedMessage must not be null"); + Assert.notNull(healthCheckFailedMessage, "'healthCheckFailedMessage' must not be null"); this.healthCheckFailedMessage = healthCheckFailedMessage; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AdditionalHealthEndpointPath.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AdditionalHealthEndpointPath.java index 2b3cccd09695..2ac395be5057 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AdditionalHealthEndpointPath.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/AdditionalHealthEndpointPath.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -103,10 +103,10 @@ public String toString() { * @return the new instance */ public static AdditionalHealthEndpointPath from(String value) { - Assert.hasText(value, "Value must not be null"); + Assert.hasText(value, "'value' must not be null"); String[] values = value.split(":"); - Assert.isTrue(values.length == 2, "Value must contain a valid namespace and value separated by ':'."); - Assert.isTrue(StringUtils.hasText(values[0]), "Value must contain a valid namespace."); + Assert.isTrue(values.length == 2, "'value' must contain a valid namespace and value separated by ':'."); + Assert.isTrue(StringUtils.hasText(values[0]), "'value' must contain a valid namespace."); WebServerNamespace namespace = WebServerNamespace.from(values[0]); validateValue(values[1]); return new AdditionalHealthEndpointPath(namespace, values[1]); @@ -120,15 +120,15 @@ public static AdditionalHealthEndpointPath from(String value) { * @return the new instance */ public static AdditionalHealthEndpointPath of(WebServerNamespace webServerNamespace, String value) { - Assert.notNull(webServerNamespace, "The server namespace must not be null."); - Assert.notNull(value, "The value must not be null."); + Assert.notNull(webServerNamespace, "'webServerNamespace' must not be null."); + Assert.notNull(value, "'value' must not be null."); validateValue(value); return new AdditionalHealthEndpointPath(webServerNamespace, value); } private static void validateValue(String value) { Assert.isTrue(StringUtils.countOccurrencesOf(value, "/") <= 1 && value.indexOf("/") <= 0, - "Value must contain only one segment."); + "'value' must contain only one segment."); } } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/CompositeHealth.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/CompositeHealth.java index 6f1a8b7c247c..94e808aae59d 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/CompositeHealth.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/CompositeHealth.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -44,7 +44,7 @@ public class CompositeHealth extends HealthComponent { private final Map details; CompositeHealth(ApiVersion apiVersion, Status status, Map components) { - Assert.notNull(status, "Status must not be null"); + Assert.notNull(status, "'status' must not be null"); this.status = status; this.components = (apiVersion != ApiVersion.V3) ? null : sort(components); this.details = (apiVersion != ApiVersion.V2) ? null : sort(components); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/CompositeHealthContributorReactiveAdapter.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/CompositeHealthContributorReactiveAdapter.java index 41dace35eec9..4cb6560332d0 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/CompositeHealthContributorReactiveAdapter.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/CompositeHealthContributorReactiveAdapter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,7 +33,7 @@ class CompositeHealthContributorReactiveAdapter implements CompositeReactiveHeal private final CompositeHealthContributor delegate; CompositeHealthContributorReactiveAdapter(CompositeHealthContributor delegate) { - Assert.notNull(delegate, "Delegate must not be null"); + Assert.notNull(delegate, "'delegate' must not be null"); this.delegate = delegate; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/DefaultContributorRegistry.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/DefaultContributorRegistry.java index faae11ae445b..7e209c2ce070 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/DefaultContributorRegistry.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/DefaultContributorRegistry.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -50,8 +50,8 @@ class DefaultContributorRegistry implements ContributorRegistry { } DefaultContributorRegistry(Map contributors, Function nameFactory) { - Assert.notNull(contributors, "Contributors must not be null"); - Assert.notNull(nameFactory, "NameFactory must not be null"); + Assert.notNull(contributors, "'contributors' must not be null"); + Assert.notNull(nameFactory, "'nameFactory' must not be null"); this.nameFactory = nameFactory; Map namedContributors = new LinkedHashMap<>(); contributors.forEach((name, contributor) -> namedContributors.put(nameFactory.apply(name), contributor)); @@ -60,8 +60,8 @@ class DefaultContributorRegistry implements ContributorRegistry { @Override public void registerContributor(String name, C contributor) { - Assert.notNull(name, "Name must not be null"); - Assert.notNull(contributor, "Contributor must not be null"); + Assert.notNull(name, "'name' must not be null"); + Assert.notNull(contributor, "'contributor' must not be null"); String adaptedName = this.nameFactory.apply(name); synchronized (this.monitor) { Assert.state(!this.contributors.containsKey(adaptedName), @@ -74,7 +74,7 @@ public void registerContributor(String name, C contributor) { @Override public C unregisterContributor(String name) { - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); String adaptedName = this.nameFactory.apply(name); synchronized (this.monitor) { C unregistered = this.contributors.get(adaptedName); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/Health.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/Health.java index b0a063414fcc..ec50576cbc4a 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/Health.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/Health.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -60,7 +60,7 @@ public final class Health extends HealthComponent { * @param builder the Builder to use */ private Health(Builder builder) { - Assert.notNull(builder, "Builder must not be null"); + Assert.notNull(builder, "'builder' must not be null"); this.status = builder.status; this.details = Collections.unmodifiableMap(builder.details); } @@ -207,7 +207,7 @@ public Builder() { * @param status the {@link Status} to use */ public Builder(Status status) { - Assert.notNull(status, "Status must not be null"); + Assert.notNull(status, "'status' must not be null"); this.status = status; this.details = new LinkedHashMap<>(); } @@ -219,21 +219,21 @@ public Builder(Status status) { * @param details the details {@link Map} to use */ public Builder(Status status, Map details) { - Assert.notNull(status, "Status must not be null"); - Assert.notNull(details, "Details must not be null"); + Assert.notNull(status, "'status' must not be null"); + Assert.notNull(details, "'details' must not be null"); this.status = status; this.details = new LinkedHashMap<>(details); } /** * Record detail for given {@link Exception}. - * @param ex the exception + * @param exception the exception * @return this {@link Builder} instance */ - public Builder withException(Throwable ex) { - Assert.notNull(ex, "Exception must not be null"); - this.exception = ex; - return withDetail("error", ex.getClass().getName() + ": " + ex.getMessage()); + public Builder withException(Throwable exception) { + Assert.notNull(exception, "'exception' must not be null"); + this.exception = exception; + return withDetail("error", exception.getClass().getName() + ": " + exception.getMessage()); } /** @@ -243,8 +243,8 @@ public Builder withException(Throwable ex) { * @return this {@link Builder} instance */ public Builder withDetail(String key, Object value) { - Assert.notNull(key, "Key must not be null"); - Assert.notNull(value, "Value must not be null"); + Assert.notNull(key, "'key' must not be null"); + Assert.notNull(value, "'value' must not be null"); this.details.put(key, value); return this; } @@ -257,7 +257,7 @@ public Builder withDetail(String key, Object value) { * @since 2.1.0 */ public Builder withDetails(Map details) { - Assert.notNull(details, "Details must not be null"); + Assert.notNull(details, "'details' must not be null"); this.details.putAll(details); return this; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthEndpointGroups.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthEndpointGroups.java index 87e3deb4935e..87170a24dd4b 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthEndpointGroups.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthEndpointGroups.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -58,7 +58,7 @@ public interface HealthEndpointGroups { * @since 2.6.0 */ default HealthEndpointGroup get(AdditionalHealthEndpointPath path) { - Assert.notNull(path, "Path must not be null"); + Assert.notNull(path, "'path' must not be null"); for (String name : getNames()) { HealthEndpointGroup group = get(name); if (path.equals(group.getAdditionalPath())) { @@ -76,7 +76,7 @@ default HealthEndpointGroup get(AdditionalHealthEndpointPath path) { * @since 2.6.0 */ default Set getAllWithAdditionalPath(WebServerNamespace namespace) { - Assert.notNull(namespace, "Namespace must not be null"); + Assert.notNull(namespace, "'namespace' must not be null"); Set filteredGroups = new LinkedHashSet<>(); getNames().stream() .map(this::get) @@ -92,8 +92,8 @@ default Set getAllWithAdditionalPath(WebServerNamespace nam * @return a new {@link HealthEndpointGroups} instance */ static HealthEndpointGroups of(HealthEndpointGroup primary, Map additional) { - Assert.notNull(primary, "Primary must not be null"); - Assert.notNull(additional, "Additional must not be null"); + Assert.notNull(primary, "'primary' must not be null"); + Assert.notNull(additional, "'additional' must not be null"); return new HealthEndpointGroups() { @Override diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthEndpointSupport.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthEndpointSupport.java index 79445877bd62..c3b30aa35918 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthEndpointSupport.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthEndpointSupport.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -63,8 +63,8 @@ abstract class HealthEndpointSupport { */ HealthEndpointSupport(ContributorRegistry registry, HealthEndpointGroups groups, Duration slowIndicatorLoggingThreshold) { - Assert.notNull(registry, "Registry must not be null"); - Assert.notNull(groups, "Groups must not be null"); + Assert.notNull(registry, "'registry' must not be null"); + Assert.notNull(groups, "'groups' must not be null"); this.registry = registry; this.groups = groups; this.slowIndicatorLoggingThreshold = slowIndicatorLoggingThreshold; diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthIndicatorReactiveAdapter.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthIndicatorReactiveAdapter.java index 0e9609bda449..ae231c55be7e 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthIndicatorReactiveAdapter.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/HealthIndicatorReactiveAdapter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ class HealthIndicatorReactiveAdapter implements ReactiveHealthIndicator { private final HealthIndicator delegate; HealthIndicatorReactiveAdapter(HealthIndicator delegate) { - Assert.notNull(delegate, "Delegate must not be null"); + Assert.notNull(delegate, "'delegate' must not be null"); this.delegate = delegate; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/NamedContributor.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/NamedContributor.java index 3922eeb4e92e..9bf96859d382 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/NamedContributor.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/NamedContributor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,8 +42,8 @@ public interface NamedContributor { C getContributor(); static NamedContributor of(String name, C contributor) { - Assert.notNull(name, "Name must not be null"); - Assert.notNull(contributor, "Contributor must not be null"); + Assert.notNull(name, "'name' must not be null"); + Assert.notNull(contributor, "'contributor' must not be null"); return new NamedContributor<>() { @Override diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/NamedContributorsMapAdapter.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/NamedContributorsMapAdapter.java index e8ed933cbbac..833d13a2884f 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/NamedContributorsMapAdapter.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/NamedContributorsMapAdapter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,24 +40,25 @@ abstract class NamedContributorsMapAdapter implements NamedContributors private final Map map; NamedContributorsMapAdapter(Map map, Function valueAdapter) { - Assert.notNull(map, "Map must not be null"); - Assert.notNull(valueAdapter, "ValueAdapter must not be null"); - map.keySet().forEach(this::validateKey); - this.map = Collections.unmodifiableMap(map.entrySet() - .stream() - .collect(LinkedHashMap::new, - (result, entry) -> result.put(entry.getKey(), adapt(entry.getValue(), valueAdapter)), Map::putAll)); + Assert.notNull(map, "'map' must not be null"); + Assert.notNull(valueAdapter, "'valueAdapter' must not be null"); + map.keySet().forEach(this::validateMapKey); + this.map = Collections.unmodifiableMap(map.entrySet().stream().collect(LinkedHashMap::new, (result, entry) -> { + String key = entry.getKey(); + C value = adaptMapValue(entry.getValue(), valueAdapter); + result.put(key, value); + }, Map::putAll)); } - private void validateKey(String value) { - Assert.notNull(value, "Map must not contain null keys"); - Assert.isTrue(!value.contains("/"), "Map keys must not contain a '/'"); + private void validateMapKey(String value) { + Assert.notNull(value, "'map' must not contain null keys"); + Assert.isTrue(!value.contains("/"), "'map' keys must not contain a '/'"); } - private C adapt(V value, Function valueAdapter) { + private C adaptMapValue(V value, Function valueAdapter) { C contributor = (value != null) ? valueAdapter.apply(value) : null; - Assert.notNull(contributor, "Map must not contain null values"); + Assert.notNull(contributor, "'map' must not contain null values"); return contributor; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/ReactiveHealthContributor.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/ReactiveHealthContributor.java index 0519c5c26871..2806e1ab5437 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/ReactiveHealthContributor.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/ReactiveHealthContributor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ public interface ReactiveHealthContributor { static ReactiveHealthContributor adapt(HealthContributor healthContributor) { - Assert.notNull(healthContributor, "HealthContributor must not be null"); + Assert.notNull(healthContributor, "'healthContributor' must not be null"); if (healthContributor instanceof HealthIndicator healthIndicator) { return new HealthIndicatorReactiveAdapter(healthIndicator); } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/Status.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/Status.java index d4113e57a6fb..81ec7f655d4d 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/Status.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/health/Status.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -78,8 +78,8 @@ public Status(String code) { * @param description a description of the status */ public Status(String code, String description) { - Assert.notNull(code, "Code must not be null"); - Assert.notNull(description, "Description must not be null"); + Assert.notNull(code, "'code' must not be null"); + Assert.notNull(description, "'description' must not be null"); this.code = code; this.description = description; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/InfoEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/InfoEndpoint.java index 2c48fc18d2df..d293d6eb1413 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/InfoEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/InfoEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,7 +42,7 @@ public class InfoEndpoint { * @param infoContributors the info contributors to use */ public InfoEndpoint(List infoContributors) { - Assert.notNull(infoContributors, "Info contributors must not be null"); + Assert.notNull(infoContributors, "'infoContributors' must not be null"); this.infoContributors = infoContributors; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/ProcessInfoContributor.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/ProcessInfoContributor.java index 1a1cd3d7540b..d9e4b3ffdf16 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/ProcessInfoContributor.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/ProcessInfoContributor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,6 +17,7 @@ package org.springframework.boot.actuate.info; import org.springframework.aot.hint.BindingReflectionHintsRegistrar; +import org.springframework.aot.hint.MemberCategory; import org.springframework.aot.hint.RuntimeHints; import org.springframework.aot.hint.RuntimeHintsRegistrar; import org.springframework.boot.actuate.info.Info.Builder; @@ -51,6 +52,9 @@ static class ProcessInfoContributorRuntimeHints implements RuntimeHintsRegistrar @Override public void registerHints(RuntimeHints hints, ClassLoader classLoader) { this.bindingRegistrar.registerReflectionHints(hints.reflection(), ProcessInfo.class); + hints.reflection() + .registerTypeIfPresent(classLoader, "jdk.management.VirtualThreadSchedulerMXBean", + MemberCategory.INVOKE_PUBLIC_METHODS); } } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/SimpleInfoContributor.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/SimpleInfoContributor.java index ca841638d339..fce74e4a2918 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/SimpleInfoContributor.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/info/SimpleInfoContributor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ public class SimpleInfoContributor implements InfoContributor { private final Object detail; public SimpleInfoContributor(String prefix, Object detail) { - Assert.notNull(prefix, "Prefix must not be null"); + Assert.notNull(prefix, "'prefix' must not be null"); this.prefix = prefix; this.detail = detail; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/ldap/LdapHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/ldap/LdapHealthIndicator.java index 2000e549b62f..d02a3bbf449a 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/ldap/LdapHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/ldap/LdapHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,7 @@ public class LdapHealthIndicator extends AbstractHealthIndicator { public LdapHealthIndicator(LdapOperations ldapOperations) { super("LDAP health check failed"); - Assert.notNull(ldapOperations, "LdapOperations must not be null"); + Assert.notNull(ldapOperations, "'ldapOperations' must not be null"); this.ldapOperations = ldapOperations; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/liquibase/LiquibaseEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/liquibase/LiquibaseEndpoint.java index a7474dd18879..233b020112b0 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/liquibase/LiquibaseEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/liquibase/LiquibaseEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -51,7 +51,7 @@ public class LiquibaseEndpoint { private final ApplicationContext context; public LiquibaseEndpoint(ApplicationContext context) { - Assert.notNull(context, "Context must be specified"); + Assert.notNull(context, "'context' must be specified"); this.context = context; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/logging/LoggersEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/logging/LoggersEndpoint.java index b46eb95a8d47..dc25b1cf7f27 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/logging/LoggersEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/logging/LoggersEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -64,8 +64,8 @@ public class LoggersEndpoint { * @param loggerGroups the logger group to expose */ public LoggersEndpoint(LoggingSystem loggingSystem, LoggerGroups loggerGroups) { - Assert.notNull(loggingSystem, "LoggingSystem must not be null"); - Assert.notNull(loggerGroups, "LoggerGroups must not be null"); + Assert.notNull(loggingSystem, "'loggingSystem' must not be null"); + Assert.notNull(loggerGroups, "'loggerGroups' must not be null"); this.loggingSystem = loggingSystem; this.loggerGroups = loggerGroups; } @@ -88,7 +88,7 @@ private Map getGroups() { @ReadOperation public LoggerLevelsDescriptor loggerLevels(@Selector String name) { - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); LoggerGroup group = this.loggerGroups.get(name); if (group != null) { return new GroupLoggerLevelsDescriptor(group.getConfiguredLevel(), group.getMembers()); @@ -99,7 +99,7 @@ public LoggerLevelsDescriptor loggerLevels(@Selector String name) { @WriteOperation public void configureLogLevel(@Selector String name, @Nullable LogLevel configuredLevel) { - Assert.notNull(name, "Name must not be empty"); + Assert.notNull(name, "'name' must not be empty"); LoggerGroup group = this.loggerGroups.get(name); if (group != null && group.hasMembers()) { group.configureLogLevel(configuredLevel, this.loggingSystem::setLogLevel); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/management/HeapDumpWebEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/management/HeapDumpWebEndpoint.java index b120533b2dbe..820e345059a1 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/management/HeapDumpWebEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/management/HeapDumpWebEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -209,7 +209,7 @@ private OpenJ9DiagnosticsMXBeanHeapDumper() { @Override public File dumpHeap(Boolean live) throws IOException, InterruptedException { - Assert.isNull(live, "OpenJ9DiagnosticsMXBean does not support live parameter when dumping the heap"); + Assert.state(live == null, "OpenJ9DiagnosticsMXBean does not support live parameter when dumping the heap"); return new File( (String) ReflectionUtils.invokeMethod(this.dumpHeapMethod, this.diagnosticMXBean, "heap", null)); } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/amqp/RabbitMetrics.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/amqp/RabbitMetrics.java index 911f858e57ac..6054386f180a 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/amqp/RabbitMetrics.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/amqp/RabbitMetrics.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,7 +45,7 @@ public class RabbitMetrics implements MeterBinder { * @param tags tags to apply to all recorded metrics */ public RabbitMetrics(ConnectionFactory connectionFactory, Iterable tags) { - Assert.notNull(connectionFactory, "ConnectionFactory must not be null"); + Assert.notNull(connectionFactory, "'connectionFactory' must not be null"); this.connectionFactory = connectionFactory; this.tags = (tags != null) ? tags : Collections.emptyList(); } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusPushGatewayManager.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusPushGatewayManager.java index dc3ebe4d31ac..81d4dba3dba6 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusPushGatewayManager.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusPushGatewayManager.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,13 +17,11 @@ package org.springframework.boot.actuate.metrics.export.prometheus; import java.time.Duration; -import java.util.Map; import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.ScheduledFuture; -import io.prometheus.client.CollectorRegistry; -import io.prometheus.client.exporter.PushGateway; +import io.prometheus.metrics.exporter.pushgateway.PushGateway; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; @@ -46,12 +44,6 @@ public class PrometheusPushGatewayManager { private final PushGateway pushGateway; - private final CollectorRegistry registry; - - private final String job; - - private final Map groupingKey; - private final ShutdownOperation shutdownOperation; private final TaskScheduler scheduler; @@ -59,43 +51,24 @@ public class PrometheusPushGatewayManager { private final ScheduledFuture scheduled; /** - * Create a new {@link PrometheusPushGatewayManager} instance using a single threaded - * {@link TaskScheduler}. + * Create a new {@link PrometheusPushGatewayManager} instance. * @param pushGateway the source push gateway - * @param registry the collector registry to push * @param pushRate the rate at which push operations occur - * @param job the job ID for the operation - * @param groupingKeys an optional set of grouping keys for the operation * @param shutdownOperation the shutdown operation that should be performed when - * context is closed. + * context is closed + * @since 3.5.0 */ - public PrometheusPushGatewayManager(PushGateway pushGateway, CollectorRegistry registry, Duration pushRate, - String job, Map groupingKeys, ShutdownOperation shutdownOperation) { - this(pushGateway, registry, new PushGatewayTaskScheduler(), pushRate, job, groupingKeys, shutdownOperation); + public PrometheusPushGatewayManager(PushGateway pushGateway, Duration pushRate, + ShutdownOperation shutdownOperation) { + this(pushGateway, new PushGatewayTaskScheduler(), pushRate, shutdownOperation); } - /** - * Create a new {@link PrometheusPushGatewayManager} instance. - * @param pushGateway the source push gateway - * @param registry the collector registry to push - * @param scheduler the scheduler used for operations - * @param pushRate the rate at which push operations occur - * @param job the job ID for the operation - * @param groupingKey an optional set of grouping keys for the operation - * @param shutdownOperation the shutdown operation that should be performed when - * context is closed. - */ - public PrometheusPushGatewayManager(PushGateway pushGateway, CollectorRegistry registry, TaskScheduler scheduler, - Duration pushRate, String job, Map groupingKey, ShutdownOperation shutdownOperation) { - Assert.notNull(pushGateway, "PushGateway must not be null"); - Assert.notNull(registry, "Registry must not be null"); - Assert.notNull(scheduler, "Scheduler must not be null"); - Assert.notNull(pushRate, "PushRate must not be null"); - Assert.hasLength(job, "Job must not be empty"); + PrometheusPushGatewayManager(PushGateway pushGateway, TaskScheduler scheduler, Duration pushRate, + ShutdownOperation shutdownOperation) { + Assert.notNull(pushGateway, "'pushGateway' must not be null"); + Assert.notNull(scheduler, "'scheduler' must not be null"); + Assert.notNull(pushRate, "'pushRate' must not be null"); this.pushGateway = pushGateway; - this.registry = registry; - this.job = job; - this.groupingKey = groupingKey; this.shutdownOperation = (shutdownOperation != null) ? shutdownOperation : ShutdownOperation.NONE; this.scheduler = scheduler; this.scheduled = this.scheduler.scheduleAtFixedRate(this::post, pushRate); @@ -103,7 +76,7 @@ public PrometheusPushGatewayManager(PushGateway pushGateway, CollectorRegistry r private void post() { try { - this.pushGateway.pushAdd(this.registry, this.job, this.groupingKey); + this.pushGateway.pushAdd(); } catch (Throwable ex) { logger.warn("Unexpected exception thrown by POST of metrics to Prometheus Pushgateway", ex); @@ -112,7 +85,7 @@ private void post() { private void put() { try { - this.pushGateway.push(this.registry, this.job, this.groupingKey); + this.pushGateway.push(); } catch (Throwable ex) { logger.warn("Unexpected exception thrown by PUT of metrics to Prometheus Pushgateway", ex); @@ -121,7 +94,7 @@ private void put() { private void delete() { try { - this.pushGateway.delete(this.job, this.groupingKey); + this.pushGateway.delete(); } catch (Throwable ex) { logger.warn("Unexpected exception thrown by DELETE of metrics from Prometheus Pushgateway", ex); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusScrapeEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusScrapeEndpoint.java index 2e795433b5b6..1f925949ddc5 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusScrapeEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusScrapeEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,17 +53,6 @@ public class PrometheusScrapeEndpoint { private volatile int nextMetricsScrapeSize = 16; - /** - * Creates a new {@link PrometheusScrapeEndpoint}. - * @param prometheusRegistry the Prometheus registry to use - * @deprecated since 3.3.1 for removal in 3.5.0 in favor of - * {@link #PrometheusScrapeEndpoint(PrometheusRegistry, Properties)} - */ - @Deprecated(since = "3.3.1", forRemoval = true) - public PrometheusScrapeEndpoint(PrometheusRegistry prometheusRegistry) { - this(prometheusRegistry, null); - } - /** * Creates a new {@link PrometheusScrapeEndpoint}. * @param prometheusRegistry the Prometheus registry to use diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusSimpleclientScrapeEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusSimpleclientScrapeEndpoint.java deleted file mode 100644 index 7e81405cdd26..000000000000 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusSimpleclientScrapeEndpoint.java +++ /dev/null @@ -1,77 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.metrics.export.prometheus; - -import java.io.IOException; -import java.io.StringWriter; -import java.io.Writer; -import java.util.Enumeration; -import java.util.Set; - -import io.prometheus.client.Collector.MetricFamilySamples; -import io.prometheus.client.CollectorRegistry; - -import org.springframework.boot.actuate.endpoint.annotation.Endpoint; -import org.springframework.boot.actuate.endpoint.annotation.ReadOperation; -import org.springframework.boot.actuate.endpoint.web.WebEndpointResponse; -import org.springframework.boot.actuate.endpoint.web.annotation.WebEndpoint; -import org.springframework.lang.Nullable; - -/** - * {@link Endpoint @Endpoint} that uses the Prometheus simpleclient to output metrics in a - * format that can be scraped by the Prometheus server. - * - * @author Jon Schneider - * @author Johnny Lim - * @since 2.0.0 - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of - * {@link PrometheusScrapeEndpoint} - */ -@Deprecated(since = "3.3.0", forRemoval = true) -@WebEndpoint(id = "prometheus") -public class PrometheusSimpleclientScrapeEndpoint { - - private static final int METRICS_SCRAPE_CHARS_EXTRA = 1024; - - private final CollectorRegistry collectorRegistry; - - private volatile int nextMetricsScrapeSize = 16; - - public PrometheusSimpleclientScrapeEndpoint(CollectorRegistry collectorRegistry) { - this.collectorRegistry = collectorRegistry; - } - - @SuppressWarnings("removal") - @ReadOperation(producesFrom = TextOutputFormat.class) - public WebEndpointResponse scrape(TextOutputFormat format, @Nullable Set includedNames) { - try { - Writer writer = new StringWriter(this.nextMetricsScrapeSize); - Enumeration samples = (includedNames != null) - ? this.collectorRegistry.filteredMetricFamilySamples(includedNames) - : this.collectorRegistry.metricFamilySamples(); - format.write(writer, samples); - String scrapePage = writer.toString(); - this.nextMetricsScrapeSize = scrapePage.length() + METRICS_SCRAPE_CHARS_EXTRA; - return new WebEndpointResponse<>(scrapePage, format); - } - catch (IOException ex) { - // This actually never happens since StringWriter doesn't throw an IOException - throw new IllegalStateException("Writing metrics failed", ex); - } - } - -} diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/TextOutputFormat.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/TextOutputFormat.java deleted file mode 100644 index 659893d8f298..000000000000 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/export/prometheus/TextOutputFormat.java +++ /dev/null @@ -1,82 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.metrics.export.prometheus; - -import java.io.IOException; -import java.io.Writer; -import java.util.Enumeration; - -import io.prometheus.client.Collector.MetricFamilySamples; -import io.prometheus.client.exporter.common.TextFormat; - -import org.springframework.boot.actuate.endpoint.Producible; -import org.springframework.util.MimeType; -import org.springframework.util.MimeTypeUtils; - -/** - * A {@link Producible} enum for supported Prometheus {@link TextFormat}. - * - * @author Andy Wilkinson - * @since 2.5.0 - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of {@link PrometheusOutputFormat} - */ -@Deprecated(since = "3.3.0", forRemoval = true) -public enum TextOutputFormat implements Producible { - - /** - * Prometheus text version 0.0.4. - */ - CONTENT_TYPE_004(TextFormat.CONTENT_TYPE_004) { - - @Override - void write(Writer writer, Enumeration samples) throws IOException { - TextFormat.write004(writer, samples); - } - - @Override - public boolean isDefault() { - return true; - } - - }, - - /** - * OpenMetrics text version 1.0.0. - */ - CONTENT_TYPE_OPENMETRICS_100(TextFormat.CONTENT_TYPE_OPENMETRICS_100) { - - @Override - void write(Writer writer, Enumeration samples) throws IOException { - TextFormat.writeOpenMetrics100(writer, samples); - } - - }; - - private final MimeType mimeType; - - TextOutputFormat(String mimeType) { - this.mimeType = MimeTypeUtils.parseMimeType(mimeType); - } - - @Override - public MimeType getProducedMimeType() { - return this.mimeType; - } - - abstract void write(Writer writer, Enumeration samples) throws IOException; - -} diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/jdbc/DataSourcePoolMetrics.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/jdbc/DataSourcePoolMetrics.java index f22023a1ef24..afa0efc3603e 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/jdbc/DataSourcePoolMetrics.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/jdbc/DataSourcePoolMetrics.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,8 +56,8 @@ public DataSourcePoolMetrics(DataSource dataSource, Collection tags) { - Assert.notNull(dataSource, "DataSource must not be null"); - Assert.notNull(metadataProvider, "MetadataProvider must not be null"); + Assert.notNull(dataSource, "'dataSource' must not be null"); + Assert.notNull(metadataProvider, "'metadataProvider' must not be null"); this.dataSource = dataSource; this.metadataProvider = new CachingDataSourcePoolMetadataProvider(metadataProvider); this.tags = Tags.concat(tags, "name", name); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/system/DiskSpaceMetricsBinder.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/system/DiskSpaceMetricsBinder.java index d2bcf25ce0d4..e19a43f93f24 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/system/DiskSpaceMetricsBinder.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/system/DiskSpaceMetricsBinder.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,7 @@ public class DiskSpaceMetricsBinder implements MeterBinder { private final Iterable tags; public DiskSpaceMetricsBinder(List paths, Iterable tags) { - Assert.notEmpty(paths, "Paths must not be empty"); + Assert.notEmpty(paths, "'paths' must not be empty"); this.paths = paths; this.tags = tags; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/web/client/ObservationRestClientCustomizer.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/web/client/ObservationRestClientCustomizer.java index 904e94260340..7ae543714d80 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/web/client/ObservationRestClientCustomizer.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/metrics/web/client/ObservationRestClientCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,8 +43,8 @@ public class ObservationRestClientCustomizer implements RestClientCustomizer { */ public ObservationRestClientCustomizer(ObservationRegistry observationRegistry, ClientRequestObservationConvention observationConvention) { - Assert.notNull(observationConvention, "ObservationConvention must not be null"); - Assert.notNull(observationRegistry, "ObservationRegistry must not be null"); + Assert.notNull(observationConvention, "'observationConvention' must not be null"); + Assert.notNull(observationRegistry, "'observationRegistry' must not be null"); this.observationRegistry = observationRegistry; this.observationConvention = observationConvention; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/quartz/QuartzEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/quartz/QuartzEndpoint.java index e6cdde6920d2..67b906dcb010 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/quartz/QuartzEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/quartz/QuartzEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,6 +17,7 @@ package org.springframework.boot.actuate.quartz; import java.time.Duration; +import java.time.Instant; import java.time.LocalTime; import java.time.temporal.ChronoUnit; import java.time.temporal.TemporalUnit; @@ -75,7 +76,7 @@ public class QuartzEndpoint { private final Sanitizer sanitizer; public QuartzEndpoint(Scheduler scheduler, Iterable sanitizingFunctions) { - Assert.notNull(scheduler, "Scheduler must not be null"); + Assert.notNull(scheduler, "'scheduler' must not be null"); this.scheduler = scheduler; this.sanitizer = new Sanitizer(sanitizingFunctions); } @@ -202,14 +203,34 @@ public QuartzJobDetailsDescriptor quartzJob(String groupName, String jobName, bo throws SchedulerException { JobKey jobKey = JobKey.jobKey(jobName, groupName); JobDetail jobDetail = this.scheduler.getJobDetail(jobKey); - if (jobDetail != null) { - List triggers = this.scheduler.getTriggersOfJob(jobKey); - return new QuartzJobDetailsDescriptor(jobDetail.getKey().getGroup(), jobDetail.getKey().getName(), - jobDetail.getDescription(), jobDetail.getJobClass().getName(), jobDetail.isDurable(), - jobDetail.requestsRecovery(), sanitizeJobDataMap(jobDetail.getJobDataMap(), showUnsanitized), - extractTriggersSummary(triggers)); + if (jobDetail == null) { + return null; } - return null; + List triggers = this.scheduler.getTriggersOfJob(jobKey); + return new QuartzJobDetailsDescriptor(jobDetail, sanitizeJobDataMap(jobDetail.getJobDataMap(), showUnsanitized), + extractTriggersSummary(triggers)); + } + + /** + * Triggers (execute it now) a Quartz job by its group and job name. + * @param groupName the name of the job's group + * @param jobName the name of the job + * @return a description of the triggered job or {@code null} if the job does not + * exist + * @throws SchedulerException if there is an error triggering the job + * @since 3.5.0 + */ + public QuartzJobTriggerDescriptor triggerQuartzJob(String groupName, String jobName) throws SchedulerException { + return triggerQuartzJob(JobKey.jobKey(jobName, groupName)); + } + + private QuartzJobTriggerDescriptor triggerQuartzJob(JobKey jobKey) throws SchedulerException { + JobDetail jobDetail = this.scheduler.getJobDetail(jobKey); + if (jobDetail == null) { + return null; + } + this.scheduler.triggerJob(jobKey); + return new QuartzJobTriggerDescriptor(jobDetail); } private static List> extractTriggersSummary(List triggers) { @@ -258,12 +279,12 @@ private static LocalTime getLocalTime(TimeOfDay timeOfDay) { } private Map sanitizeJobDataMap(JobDataMap dataMap, boolean showUnsanitized) { - if (dataMap != null) { - Map map = new LinkedHashMap<>(dataMap.getWrappedMap()); - map.replaceAll((key, value) -> getSanitizedValue(showUnsanitized, key, value)); - return map; + if (dataMap == null) { + return null; } - return null; + Map map = new LinkedHashMap<>(dataMap.getWrappedMap()); + map.replaceAll((key, value) -> getSanitizedValue(showUnsanitized, key, value)); + return map; } private Object getSanitizedValue(boolean showUnsanitized, String key, Object value) { @@ -351,7 +372,7 @@ public static final class QuartzJobGroupSummaryDescriptor implements OperationRe private final Map jobs; - private QuartzJobGroupSummaryDescriptor(String group, Map jobs) { + QuartzJobGroupSummaryDescriptor(String group, Map jobs) { this.group = group; this.jobs = jobs; } @@ -373,7 +394,7 @@ public static final class QuartzJobSummaryDescriptor { private final String className; - private QuartzJobSummaryDescriptor(JobDetail job) { + QuartzJobSummaryDescriptor(JobDetail job) { this.className = job.getJobClass().getName(); } @@ -387,6 +408,46 @@ public String getClassName() { } + /** + * Description of a triggered on-demand {@link Job Quartz Job}. + * + * @since 3.5.0 + */ + public static final class QuartzJobTriggerDescriptor { + + private final String group; + + private final String name; + + private final String className; + + private final Instant triggerTime; + + QuartzJobTriggerDescriptor(JobDetail jobDetail) { + this.group = jobDetail.getKey().getGroup(); + this.name = jobDetail.getKey().getName(); + this.className = jobDetail.getJobClass().getName(); + this.triggerTime = Instant.now(); + } + + public String getGroup() { + return this.group; + } + + public String getName() { + return this.name; + } + + public String getClassName() { + return this.className; + } + + public Instant getTriggerTime() { + return this.triggerTime; + } + + } + /** * Description of a {@link Job Quartz Job}. */ @@ -408,14 +469,13 @@ public static final class QuartzJobDetailsDescriptor implements OperationRespons private final List> triggers; - QuartzJobDetailsDescriptor(String group, String name, String description, String className, boolean durable, - boolean requestRecovery, Map data, List> triggers) { - this.group = group; - this.name = name; - this.description = description; - this.className = className; - this.durable = durable; - this.requestRecovery = requestRecovery; + QuartzJobDetailsDescriptor(JobDetail jobDetail, Map data, List> triggers) { + this.group = jobDetail.getKey().getGroup(); + this.name = jobDetail.getKey().getName(); + this.description = jobDetail.getDescription(); + this.className = jobDetail.getJobClass().getName(); + this.durable = jobDetail.isDurable(); + this.requestRecovery = jobDetail.requestsRecovery(); this.data = data; this.triggers = triggers; } @@ -465,7 +525,7 @@ public static final class QuartzTriggerGroupSummaryDescriptor implements Operati private final Triggers triggers; - private QuartzTriggerGroupSummaryDescriptor(String group, boolean paused, + QuartzTriggerGroupSummaryDescriptor(String group, boolean paused, Map> descriptionsByType) { this.group = group; this.paused = paused; @@ -497,7 +557,7 @@ public static final class Triggers { private final Map custom; - private Triggers(Map> descriptionsByType) { + Triggers(Map> descriptionsByType) { this.cron = descriptionsByType.getOrDefault(TriggerType.CRON, Collections.emptyMap()); this.dailyTimeInterval = descriptionsByType.getOrDefault(TriggerType.DAILY_INTERVAL, Collections.emptyMap()); @@ -560,30 +620,23 @@ public String getId() { */ public abstract static class TriggerDescriptor { - private static final Map, Function> DESCRIBERS = new LinkedHashMap<>(); + private static final Map, Function> DESCRIBERS; static { - DESCRIBERS.put(CronTrigger.class, (trigger) -> new CronTriggerDescriptor((CronTrigger) trigger)); - DESCRIBERS.put(SimpleTrigger.class, (trigger) -> new SimpleTriggerDescriptor((SimpleTrigger) trigger)); - DESCRIBERS.put(DailyTimeIntervalTrigger.class, + Map, Function> descriptors = new LinkedHashMap<>(); + descriptors.put(CronTrigger.class, (trigger) -> new CronTriggerDescriptor((CronTrigger) trigger)); + descriptors.put(SimpleTrigger.class, (trigger) -> new SimpleTriggerDescriptor((SimpleTrigger) trigger)); + descriptors.put(DailyTimeIntervalTrigger.class, (trigger) -> new DailyTimeIntervalTriggerDescriptor((DailyTimeIntervalTrigger) trigger)); - DESCRIBERS.put(CalendarIntervalTrigger.class, + descriptors.put(CalendarIntervalTrigger.class, (trigger) -> new CalendarIntervalTriggerDescriptor((CalendarIntervalTrigger) trigger)); + DESCRIBERS = Map.copyOf(descriptors); } private final Trigger trigger; private final TriggerType type; - private static TriggerDescriptor of(Trigger trigger) { - return DESCRIBERS.entrySet() - .stream() - .filter((entry) -> entry.getKey().isInstance(trigger)) - .map((entry) -> entry.getValue().apply(trigger)) - .findFirst() - .orElse(new CustomTriggerDescriptor(trigger)); - } - protected TriggerDescriptor(Trigger trigger, TriggerType type) { this.trigger = trigger; this.type = type; @@ -661,6 +714,15 @@ protected TriggerType getType() { return this.type; } + static TriggerDescriptor of(Trigger trigger) { + return DESCRIBERS.entrySet() + .stream() + .filter((entry) -> entry.getKey().isInstance(trigger)) + .map((entry) -> entry.getValue().apply(trigger)) + .findFirst() + .orElse(new CustomTriggerDescriptor(trigger)); + } + } /** diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/quartz/QuartzEndpointWebExtension.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/quartz/QuartzEndpointWebExtension.java index c5d3ac3e0d0f..118837adf78a 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/quartz/QuartzEndpointWebExtension.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/quartz/QuartzEndpointWebExtension.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,6 +27,7 @@ import org.springframework.boot.actuate.endpoint.Show; import org.springframework.boot.actuate.endpoint.annotation.ReadOperation; import org.springframework.boot.actuate.endpoint.annotation.Selector; +import org.springframework.boot.actuate.endpoint.annotation.WriteOperation; import org.springframework.boot.actuate.endpoint.web.WebEndpointResponse; import org.springframework.boot.actuate.endpoint.web.annotation.EndpointWebExtension; import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzGroupsDescriptor; @@ -79,6 +80,25 @@ public WebEndpointResponse quartzJobOrTrigger(SecurityContext securityCo () -> this.delegate.quartzTrigger(group, name, showUnsanitized)); } + /** + * Trigger a Quartz job. + * @param jobs path segment "jobs" + * @param group job's group + * @param name job name + * @param state desired state + * @return web endpoint response + * @throws SchedulerException if there is an error triggering the job + * @since 3.5.0 + */ + @WriteOperation + public WebEndpointResponse triggerQuartzJob(@Selector String jobs, @Selector String group, + @Selector String name, String state) throws SchedulerException { + if ("jobs".equals(jobs) && "running".equals(state)) { + return handleNull(this.delegate.triggerQuartzJob(group, name)); + } + return new WebEndpointResponse<>(WebEndpointResponse.STATUS_BAD_REQUEST); + } + private WebEndpointResponse handle(String jobsOrTriggers, ResponseSupplier jobAction, ResponseSupplier triggerAction) throws SchedulerException { if ("jobs".equals(jobsOrTriggers)) { @@ -91,10 +111,8 @@ private WebEndpointResponse handle(String jobsOrTriggers, ResponseSupplie } private WebEndpointResponse handleNull(T value) { - if (value != null) { - return new WebEndpointResponse<>(value); - } - return new WebEndpointResponse<>(WebEndpointResponse.STATUS_NOT_FOUND); + return (value != null) ? new WebEndpointResponse<>(value) + : new WebEndpointResponse<>(WebEndpointResponse.STATUS_NOT_FOUND); } @FunctionalInterface diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/r2dbc/ConnectionFactoryHealthIndicator.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/r2dbc/ConnectionFactoryHealthIndicator.java index d1c3d942cbaf..07a0fcd1c26f 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/r2dbc/ConnectionFactoryHealthIndicator.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/r2dbc/ConnectionFactoryHealthIndicator.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -63,7 +63,7 @@ public ConnectionFactoryHealthIndicator(ConnectionFactory connectionFactory) { * validation */ public ConnectionFactoryHealthIndicator(ConnectionFactory connectionFactory, String validationQuery) { - Assert.notNull(connectionFactory, "ConnectionFactory must not be null"); + Assert.notNull(connectionFactory, "'connectionFactory' must not be null"); this.connectionFactory = connectionFactory; this.validationQuery = validationQuery; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/sbom/SbomProperties.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/sbom/SbomProperties.java index 5d883a83de70..479920785ce8 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/sbom/SbomProperties.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/sbom/SbomProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,7 +28,7 @@ * @author Moritz Halbritter * @since 3.3.0 */ -@ConfigurationProperties(prefix = "management.endpoint.sbom") +@ConfigurationProperties("management.endpoint.sbom") public class SbomProperties { /** diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/session/ReactiveSessionsEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/session/ReactiveSessionsEndpoint.java index 9ee15d69fc02..d8e6bfe75599 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/session/ReactiveSessionsEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/session/ReactiveSessionsEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -50,7 +50,7 @@ public class ReactiveSessionsEndpoint { */ public ReactiveSessionsEndpoint(ReactiveSessionRepository sessionRepository, ReactiveFindByIndexNameSessionRepository indexedSessionRepository) { - Assert.notNull(sessionRepository, "ReactiveSessionRepository must not be null"); + Assert.notNull(sessionRepository, "'sessionRepository' must not be null"); this.sessionRepository = sessionRepository; this.indexedSessionRepository = indexedSessionRepository; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/session/SessionsEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/session/SessionsEndpoint.java index 1b89c83a1132..d8d56ef6ad8f 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/session/SessionsEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/session/SessionsEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,17 +42,6 @@ public class SessionsEndpoint { private final FindByIndexNameSessionRepository indexedSessionRepository; - /** - * Create a new {@link SessionsEndpoint} instance. - * @param sessionRepository the session repository - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of - * {@link #SessionsEndpoint(SessionRepository, FindByIndexNameSessionRepository)} - */ - @Deprecated(since = "3.3.0", forRemoval = true) - public SessionsEndpoint(FindByIndexNameSessionRepository sessionRepository) { - this(sessionRepository, sessionRepository); - } - /** * Create a new {@link SessionsEndpoint} instance. * @param sessionRepository the session repository @@ -61,7 +50,7 @@ public SessionsEndpoint(FindByIndexNameSessionRepository sess */ public SessionsEndpoint(SessionRepository sessionRepository, FindByIndexNameSessionRepository indexedSessionRepository) { - Assert.notNull(sessionRepository, "SessionRepository must not be null"); + Assert.notNull(sessionRepository, "'sessionRepository' must not be null"); this.sessionRepository = sessionRepository; this.indexedSessionRepository = indexedSessionRepository; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/exchanges/HttpExchangesEndpoint.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/exchanges/HttpExchangesEndpoint.java index a68370932767..321e660ef204 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/exchanges/HttpExchangesEndpoint.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/exchanges/HttpExchangesEndpoint.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,7 +40,7 @@ public class HttpExchangesEndpoint { * @param repository the exchange repository */ public HttpExchangesEndpoint(HttpExchangeRepository repository) { - Assert.notNull(repository, "Repository must not be null"); + Assert.notNull(repository, "'repository' must not be null"); this.repository = repository; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/DispatcherServletMappingDetails.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/DispatcherServletMappingDetails.java index 699ff05b4601..786ddc5519db 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/DispatcherServletMappingDetails.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/DispatcherServletMappingDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,12 +23,15 @@ * Details of a {@link DispatcherServlet} mapping. * * @author Andy Wilkinson + * @author Xiong Tang * @since 2.0.0 */ public class DispatcherServletMappingDetails { private HandlerMethodDescription handlerMethod; + private HandlerFunctionDescription handlerFunction; + private RequestMappingConditionsDescription requestMappingConditions; public HandlerMethodDescription getHandlerMethod() { @@ -39,6 +42,14 @@ void setHandlerMethod(HandlerMethodDescription handlerMethod) { this.handlerMethod = handlerMethod; } + public HandlerFunctionDescription getHandlerFunction() { + return this.handlerFunction; + } + + void setHandlerFunction(HandlerFunctionDescription handlerFunction) { + this.handlerFunction = handlerFunction; + } + public RequestMappingConditionsDescription getRequestMappingConditions() { return this.requestMappingConditions; } diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/DispatcherServletsMappingDescriptionProvider.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/DispatcherServletsMappingDescriptionProvider.java index ffb48304ee90..2225550c1bb4 100644 --- a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/DispatcherServletsMappingDescriptionProvider.java +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/DispatcherServletsMappingDescriptionProvider.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,6 +23,8 @@ import java.util.List; import java.util.Map; import java.util.Map.Entry; +import java.util.Optional; +import java.util.function.Function; import java.util.stream.Stream; import jakarta.servlet.Servlet; @@ -36,10 +38,17 @@ import org.springframework.boot.web.servlet.ServletRegistrationBean; import org.springframework.context.ApplicationContext; import org.springframework.context.annotation.ImportRuntimeHints; +import org.springframework.core.io.Resource; import org.springframework.web.context.WebApplicationContext; import org.springframework.web.method.HandlerMethod; import org.springframework.web.servlet.DispatcherServlet; import org.springframework.web.servlet.HandlerMapping; +import org.springframework.web.servlet.function.HandlerFunction; +import org.springframework.web.servlet.function.RequestPredicate; +import org.springframework.web.servlet.function.RouterFunction; +import org.springframework.web.servlet.function.RouterFunctions.Visitor; +import org.springframework.web.servlet.function.ServerRequest; +import org.springframework.web.servlet.function.support.RouterFunctionMapping; import org.springframework.web.servlet.handler.AbstractUrlHandlerMapping; import org.springframework.web.servlet.mvc.method.RequestMappingInfo; import org.springframework.web.servlet.mvc.method.RequestMappingInfoHandlerMapping; @@ -51,6 +60,7 @@ * * @author Andy Wilkinson * @author Stephane Nicoll + * @author Xiong Tang * @since 2.0.0 */ @ImportRuntimeHints(DispatcherServletsMappingDescriptionProviderRuntimeHints.class) @@ -63,6 +73,7 @@ public class DispatcherServletsMappingDescriptionProvider implements MappingDesc providers.add(new RequestMappingInfoHandlerMappingDescriptionProvider()); providers.add(new UrlHandlerMappingDescriptionProvider()); providers.add(new IterableDelegatesHandlerMappingDescriptionProvider(new ArrayList<>(providers))); + providers.add(new RouterFunctionMappingDescriptionProvider()); descriptionProviders = Collections.unmodifiableList(providers); } @@ -200,6 +211,62 @@ public List describe(Iterable handlerMappin } + private static final class RouterFunctionMappingDescriptionProvider + implements HandlerMappingDescriptionProvider { + + @Override + public Class getMappingClass() { + return RouterFunctionMapping.class; + } + + @Override + public List describe(RouterFunctionMapping handlerMapping) { + MappingDescriptionVisitor visitor = new MappingDescriptionVisitor(); + RouterFunction routerFunction = handlerMapping.getRouterFunction(); + if (routerFunction != null) { + routerFunction.accept(visitor); + } + return visitor.descriptions; + } + + } + + private static final class MappingDescriptionVisitor implements Visitor { + + private final List descriptions = new ArrayList<>(); + + @Override + public void startNested(RequestPredicate predicate) { + } + + @Override + public void endNested(RequestPredicate predicate) { + } + + @Override + public void route(RequestPredicate predicate, HandlerFunction handlerFunction) { + DispatcherServletMappingDetails details = new DispatcherServletMappingDetails(); + details.setHandlerFunction(new HandlerFunctionDescription(handlerFunction)); + this.descriptions.add( + new DispatcherServletMappingDescription(predicate.toString(), handlerFunction.toString(), details)); + } + + @Override + public void resources(Function> lookupFunction) { + + } + + @Override + public void attributes(Map attributes) { + } + + @Override + public void unknown(RouterFunction routerFunction) { + + } + + } + static class DispatcherServletsMappingDescriptionProviderRuntimeHints implements RuntimeHintsRegistrar { private final BindingReflectionHintsRegistrar bindingRegistrar = new BindingReflectionHintsRegistrar(); diff --git a/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/HandlerFunctionDescription.java b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/HandlerFunctionDescription.java new file mode 100644 index 000000000000..f62a4f55f4e3 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator/src/main/java/org/springframework/boot/actuate/web/mappings/servlet/HandlerFunctionDescription.java @@ -0,0 +1,45 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.web.mappings.servlet; + +import org.springframework.web.servlet.function.HandlerFunction; + +/** + * Description of a {@link HandlerFunction}. + * + * @author Xiong Tang + * @since 3.5.0 + */ +public class HandlerFunctionDescription { + + private final String className; + + HandlerFunctionDescription(HandlerFunction handlerFunction) { + this.className = getHandlerFunctionClassName(handlerFunction); + } + + private static String getHandlerFunctionClassName(HandlerFunction handlerFunction) { + Class functionClass = handlerFunction.getClass(); + String canonicalName = functionClass.getCanonicalName(); + return (canonicalName != null) ? canonicalName : functionClass.getName(); + } + + public String getClassName() { + return this.className; + } + +} diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/amqp/RabbitHealthIndicatorTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/amqp/RabbitHealthIndicatorTests.java index 63856b2f16b8..4a10bee81566 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/amqp/RabbitHealthIndicatorTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/amqp/RabbitHealthIndicatorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,7 +53,7 @@ class RabbitHealthIndicatorTests { @Test void createWhenRabbitTemplateIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new RabbitHealthIndicator(null)) - .withMessageContaining("RabbitTemplate must not be null"); + .withMessageContaining("'rabbitTemplate' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/audit/AuditEventTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/audit/AuditEventTests.java index 97b3cd270ff4..768742024206 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/audit/AuditEventTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/audit/AuditEventTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -60,14 +60,14 @@ void nullPrincipalIsMappedToEmptyString() { void nullTimestamp() { assertThatIllegalArgumentException() .isThrownBy(() -> new AuditEvent(null, "phil", "UNKNOWN", Collections.singletonMap("a", "b"))) - .withMessageContaining("Timestamp must not be null"); + .withMessageContaining("'timestamp' must not be null"); } @Test void nullType() { assertThatIllegalArgumentException() .isThrownBy(() -> new AuditEvent("phil", null, Collections.singletonMap("a", "b"))) - .withMessageContaining("Type must not be null"); + .withMessageContaining("'type' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/audit/InMemoryAuditEventRepositoryTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/audit/InMemoryAuditEventRepositoryTests.java index cb1ab5395691..da400a148df1 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/audit/InMemoryAuditEventRepositoryTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/audit/InMemoryAuditEventRepositoryTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -63,7 +63,7 @@ void capacity() { void addNullAuditEvent() { InMemoryAuditEventRepository repository = new InMemoryAuditEventRepository(); assertThatIllegalArgumentException().isThrownBy(() -> repository.add(null)) - .withMessageContaining("AuditEvent must not be null"); + .withMessageContaining("'event' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/availability/AvailabilityStateHealthIndicatorTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/availability/AvailabilityStateHealthIndicatorTests.java index a874e4be5f41..cc40f23f6318 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/availability/AvailabilityStateHealthIndicatorTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/availability/AvailabilityStateHealthIndicatorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,6 +28,7 @@ import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; import static org.mockito.BDDMockito.given; /** @@ -46,7 +47,7 @@ void createWhenApplicationAvailabilityIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new AvailabilityStateHealthIndicator(null, LivenessState.class, (statusMappings) -> { })) - .withMessage("ApplicationAvailability must not be null"); + .withMessage("'applicationAvailability' must not be null"); } @Test @@ -54,7 +55,7 @@ void createWhenStateTypeIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy( () -> new AvailabilityStateHealthIndicator(this.applicationAvailability, null, (statusMappings) -> { })) - .withMessage("StateType must not be null"); + .withMessage("'stateType' must not be null"); } @Test @@ -62,12 +63,12 @@ void createWhenStatusMappingIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy( () -> new AvailabilityStateHealthIndicator(this.applicationAvailability, LivenessState.class, null)) - .withMessage("StatusMappings must not be null"); + .withMessage("'statusMappings' must not be null"); } @Test void createWhenStatusMappingDoesNotCoverAllEnumsThrowsException() { - assertThatIllegalArgumentException() + assertThatIllegalStateException() .isThrownBy(() -> new AvailabilityStateHealthIndicator(this.applicationAvailability, LivenessState.class, (statusMappings) -> statusMappings.add(LivenessState.CORRECT, Status.UP))) .withMessage("StatusMappings does not include BROKEN"); diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointFilteringTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointFilteringTests.java index 16d2026a7e82..119952be0147 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointFilteringTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointFilteringTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -163,13 +163,13 @@ ConfigurationPropertiesReportEndpoint endpoint() { static class BaseConfiguration { @Bean - @ConfigurationProperties(prefix = "foo.primary") + @ConfigurationProperties("foo.primary") Foo primaryFoo() { return new Foo(); } @Bean - @ConfigurationProperties(prefix = "foo.secondary") + @ConfigurationProperties("foo.secondary") Foo secondaryFoo() { return new Foo(); } @@ -190,7 +190,7 @@ public void setName(String name) { } - @ConfigurationProperties(prefix = "only.bar") + @ConfigurationProperties("only.bar") public static class Bar { private String name = "123456"; diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointMethodAnnotationsTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointMethodAnnotationsTests.java index 1abf27582253..a82b57186df6 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointMethodAnnotationsTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointMethodAnnotationsTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -89,13 +89,13 @@ ConfigurationPropertiesReportEndpoint endpoint() { } @Bean - @ConfigurationProperties(prefix = "first") + @ConfigurationProperties("first") Foo foo() { return new Foo(); } @Bean - @ConfigurationProperties(prefix = "other") + @ConfigurationProperties("other") Foo other() { return new Foo(); } @@ -112,7 +112,7 @@ ConfigurationPropertiesReportEndpoint endpoint() { } @Bean - @ConfigurationProperties(prefix = "other") + @ConfigurationProperties("other") Bar bar() { return new Bar(); } @@ -133,7 +133,7 @@ public void setName(String name) { } - @ConfigurationProperties(prefix = "test") + @ConfigurationProperties("test") public static class Bar { private String name = "654321"; diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointParentTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointParentTests.java index 1c4ae533f94e..852b99320d31 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointParentTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointParentTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -111,7 +111,7 @@ ConfigurationPropertiesReportEndpoint endpoint() { } @Bean - @ConfigurationProperties(prefix = "other") + @ConfigurationProperties("other") OtherProperties otherProperties() { return new OtherProperties(); } @@ -122,7 +122,7 @@ static class OtherProperties { } - @ConfigurationProperties(prefix = "test") + @ConfigurationProperties("test") static class TestProperties { private String myTestProperty = "654321"; diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointSerializationTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointSerializationTests.java index 3acea77a8b68..9dfe52e64803 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointSerializationTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointSerializationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -329,7 +329,7 @@ ConfigurationPropertiesReportEndpoint endpoint() { static class FooConfig { @Bean - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") Foo foo() { return new Foo(); } @@ -341,7 +341,7 @@ Foo foo() { static class SelfReferentialConfig { @Bean - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") SelfReferential foo() { return new SelfReferential(); } @@ -353,7 +353,7 @@ SelfReferential foo() { static class MetadataCycleConfig { @Bean - @ConfigurationProperties(prefix = "bar") + @ConfigurationProperties("bar") SelfReferential foo() { return new SelfReferential(); } @@ -365,7 +365,7 @@ SelfReferential foo() { static class MapConfig { @Bean - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") MapHolder foo() { return new MapHolder(); } @@ -377,7 +377,7 @@ MapHolder foo() { static class ListConfig { @Bean - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") ListHolder foo() { return new ListHolder(); } @@ -389,7 +389,7 @@ ListHolder foo() { static class MetadataMapConfig { @Bean - @ConfigurationProperties(prefix = "spam") + @ConfigurationProperties("spam") MapHolder foo() { return new MapHolder(); } @@ -401,7 +401,7 @@ MapHolder foo() { static class AddressedConfig { @Bean - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") Addressed foo() { return new Addressed(); } @@ -413,7 +413,7 @@ Addressed foo() { static class InitializedMapAndListPropertiesConfig { @Bean - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") InitializedMapAndListProperties foo() { return new InitializedMapAndListProperties(); } @@ -569,7 +569,7 @@ static class CycleConfig { @Bean // gh-11037 - @ConfigurationProperties(prefix = "cycle") + @ConfigurationProperties("cycle") Cycle cycle() { return new Cycle(); } @@ -586,7 +586,7 @@ ConfigurationPropertiesReportEndpoint endpoint() { } @Bean - @ConfigurationProperties(prefix = "test.datasource") + @ConfigurationProperties("test.datasource") HikariDataSource hikariDataSource() { return new HikariDataSource(); } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointTests.java index 8180021e5f88..d051881df596 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -457,7 +457,7 @@ static class TestPropertiesConfiguration { } - @ConfigurationProperties(prefix = "test") + @ConfigurationProperties("test") public static class TestProperties { private String dbPassword = "123456"; @@ -524,7 +524,7 @@ static class ImmutablePropertiesConfiguration { } - @ConfigurationProperties(prefix = "immutable") + @ConfigurationProperties("immutable") public static class ImmutableProperties { private final String dbPassword; @@ -574,7 +574,7 @@ static class MultiConstructorPropertiesConfiguration { } - @ConfigurationProperties(prefix = "multiconstructor") + @ConfigurationProperties("multiconstructor") public static class MultiConstructorProperties { private final String name; @@ -613,7 +613,7 @@ String hello() { } - @ConfigurationProperties(prefix = "autowired") + @ConfigurationProperties("autowired") public static class AutowiredProperties { private final String name; diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointWebIntegrationTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointWebIntegrationTests.java index 504a0496838a..d8dec3f0b883 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointWebIntegrationTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ConfigurationPropertiesReportEndpointWebIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -116,13 +116,13 @@ ConfigurationPropertiesReportEndpointWebExtension endpointWebExtension( } @Bean - @ConfigurationProperties(prefix = "com.foo") + @ConfigurationProperties("com.foo") Foo fooDotCom() { return new Foo(); } @Bean - @ConfigurationProperties(prefix = "com.bar") + @ConfigurationProperties("com.bar") Bar barDotCom() { return new Bar(); } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ValidatedConstructorBindingProperties.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ValidatedConstructorBindingProperties.java index a2f5a40ca189..e6b04fc3c53d 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ValidatedConstructorBindingProperties.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/context/properties/ValidatedConstructorBindingProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Madhura Bhave */ @Validated -@ConfigurationProperties(prefix = "validated") +@ConfigurationProperties("validated") public class ValidatedConstructorBindingProperties { private final String name; diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/elasticsearch/ElasticsearchReactiveHealthIndicatorTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/elasticsearch/ElasticsearchReactiveHealthIndicatorTests.java index 28d3583bfd6e..ad1405977d12 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/elasticsearch/ElasticsearchReactiveHealthIndicatorTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/elasticsearch/ElasticsearchReactiveHealthIndicatorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -72,7 +72,7 @@ void shutdown() throws Exception { @Test void elasticsearchIsUp() { - setupMockResponse(200, "green"); + setupMockResponse("green"); Health health = this.healthIndicator.health().block(TIMEOUT); assertThat(health.getStatus()).isEqualTo(Status.UP); assertHealthDetailsWithStatus(health.getDetails(), "green"); @@ -80,7 +80,7 @@ void elasticsearchIsUp() { @Test void elasticsearchWithYellowStatusIsUp() { - setupMockResponse(200, "yellow"); + setupMockResponse("yellow"); Health health = this.healthIndicator.health().block(TIMEOUT); assertThat(health.getStatus()).isEqualTo(Status.UP); assertHealthDetailsWithStatus(health.getDetails(), "yellow"); @@ -104,7 +104,7 @@ void elasticsearchIsDownByResponseCode() { @Test void elasticsearchIsOutOfServiceByStatus() { - setupMockResponse(200, "red"); + setupMockResponse("red"); Health health = this.healthIndicator.health().block(TIMEOUT); assertThat(health.getStatus()).isEqualTo(Status.OUT_OF_SERVICE); assertHealthDetailsWithStatus(health.getDetails(), "red"); @@ -116,10 +116,11 @@ private void assertHealthDetailsWithStatus(Map details, String s entry("active_primary_shards", 0), entry("active_shards", 0), entry("relocating_shards", 0), entry("initializing_shards", 0), entry("unassigned_shards", 0), entry("delayed_unassigned_shards", 0), entry("number_of_pending_tasks", 0), entry("number_of_in_flight_fetch", 0), - entry("task_max_waiting_in_queue_millis", 0L), entry("active_shards_percent_as_number", 100.0)); + entry("task_max_waiting_in_queue_millis", 0L), entry("active_shards_percent_as_number", 100.0), + entry("unassigned_primary_shards", 10)); } - private void setupMockResponse(int responseCode, String status) { + private void setupMockResponse(String status) { MockResponse mockResponse = new MockResponse().setBody(createJsonResult(status)) .setHeader(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE) .setHeader("X-Elastic-Product", "Elasticsearch"); @@ -133,7 +134,8 @@ private String createJsonResult(String status) { + "\"active_shards\":0,\"relocating_shards\":0,\"initializing_shards\":0," + "\"unassigned_shards\":0,\"delayed_unassigned_shards\":0," + "\"number_of_pending_tasks\":0,\"number_of_in_flight_fetch\":0," - + "\"task_max_waiting_in_queue_millis\":0,\"active_shards_percent_as_number\":100.0}", + + "\"task_max_waiting_in_queue_millis\":0,\"active_shards_percent_as_number\":100.0," + + "\"unassigned_primary_shards\": 10 }", status); } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/elasticsearch/ElasticsearchRestClientHealthIndicatorTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/elasticsearch/ElasticsearchRestClientHealthIndicatorTests.java index 223f9df9677c..618b25381ce7 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/elasticsearch/ElasticsearchRestClientHealthIndicatorTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/elasticsearch/ElasticsearchRestClientHealthIndicatorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -122,20 +122,20 @@ private void assertHealthDetailsWithStatus(Map details, String s entry("active_primary_shards", 0), entry("active_shards", 0), entry("relocating_shards", 0), entry("initializing_shards", 0), entry("unassigned_shards", 0), entry("delayed_unassigned_shards", 0), entry("number_of_pending_tasks", 0), entry("number_of_in_flight_fetch", 0), - entry("task_max_waiting_in_queue_millis", 0), entry("active_shards_percent_as_number", 100.0)); + entry("task_max_waiting_in_queue_millis", 0), entry("active_shards_percent_as_number", 100.0), + entry("unassigned_primary_shards", 10)); } private String createJsonResult(int responseCode, String status) { if (responseCode == 200) { - return String.format( - "{\"cluster_name\":\"elasticsearch\"," - + "\"status\":\"%s\",\"timed_out\":false,\"number_of_nodes\":1," - + "\"number_of_data_nodes\":1,\"active_primary_shards\":0," - + "\"active_shards\":0,\"relocating_shards\":0,\"initializing_shards\":0," - + "\"unassigned_shards\":0,\"delayed_unassigned_shards\":0," - + "\"number_of_pending_tasks\":0,\"number_of_in_flight_fetch\":0," - + "\"task_max_waiting_in_queue_millis\":0,\"active_shards_percent_as_number\":100.0}", - status); + return String.format("{\"cluster_name\":\"elasticsearch\"," + + "\"status\":\"%s\",\"timed_out\":false,\"number_of_nodes\":1," + + "\"number_of_data_nodes\":1,\"active_primary_shards\":0," + + "\"active_shards\":0,\"relocating_shards\":0,\"initializing_shards\":0," + + "\"unassigned_shards\":0,\"delayed_unassigned_shards\":0," + + "\"number_of_pending_tasks\":0,\"number_of_in_flight_fetch\":0," + + "\"task_max_waiting_in_queue_millis\":0,\"active_shards_percent_as_number\":100.0," + + "\"unassigned_primary_shards\": 10 }", status); } return "{\n \"error\": \"Server Error\",\n \"status\": " + responseCode + "\n}"; } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/EndpointIdTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/EndpointIdTests.java index bc57d0a1d05b..687010d0563d 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/EndpointIdTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/EndpointIdTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,42 +37,43 @@ class EndpointIdTests { @Test void ofWhenNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> EndpointId.of(null)) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test void ofWhenEmptyThrowsException() { - assertThatIllegalArgumentException().isThrownBy(() -> EndpointId.of("")).withMessage("Value must not be empty"); + assertThatIllegalArgumentException().isThrownBy(() -> EndpointId.of("")) + .withMessage("'value' must not be empty"); } @Test void ofWhenContainsSlashThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> EndpointId.of("foo/bar")) - .withMessage("Value must only contain valid chars"); + .withMessage("'value' must only contain valid chars"); } @Test void ofWhenContainsBackslashThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> EndpointId.of("foo\\bar")) - .withMessage("Value must only contain valid chars"); + .withMessage("'value' must only contain valid chars"); } @Test void ofWhenHasBadCharThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> EndpointId.of("foo!bar")) - .withMessage("Value must only contain valid chars"); + .withMessage("'value' must only contain valid chars"); } @Test void ofWhenStartsWithNumberThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> EndpointId.of("1foo")) - .withMessage("Value must not start with a number"); + .withMessage("'value' must not start with a number"); } @Test void ofWhenStartsWithUppercaseLetterThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> EndpointId.of("Foo")) - .withMessage("Value must not start with an uppercase letter"); + .withMessage("'value' must not start with an uppercase letter"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/InvocationContextTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/InvocationContextTests.java index 84f2a83279cb..41f59d2f53a6 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/InvocationContextTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/InvocationContextTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,13 +45,13 @@ void whenCreatedWithoutApiVersionThenResolveApiVersionReturnsLatestVersion() { @Test void createWhenSecurityContextIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new InvocationContext(null, this.arguments)) - .withMessage("SecurityContext must not be null"); + .withMessage("'securityContext' must not be null"); } @Test void createWhenArgumentsIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new InvocationContext(this.securityContext, null)) - .withMessage("Arguments must not be null"); + .withMessage("'arguments' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/SanitizerTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/SanitizerTests.java index b5c658e34428..72e04d78ecbc 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/SanitizerTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/SanitizerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -93,6 +93,15 @@ void overridingDefaultSanitizingFunction() { assertThat(sanitizer.sanitize(password, true)).isEqualTo("------"); } + @Test + void overridingDefaultSanitizingFunctionWithFiltered() { + Sanitizer sanitizer = new Sanitizer(List.of(SanitizingFunction.sanitizeValue().ifLikelySensitive())); + SanitizableData other = new SanitizableData(null, "other", "123456"); + SanitizableData password = new SanitizableData(null, "password", "123456"); + assertThat(sanitizer.sanitize(other, true)).isEqualTo("123456"); + assertThat(sanitizer.sanitize(password, true)).isEqualTo(SanitizableData.SANITIZED_VALUE); + } + @Test void whenValueSanitizedLaterSanitizingFunctionsShouldBeSkipped() { final String sameKey = "custom"; diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/SanitizingFunctionTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/SanitizingFunctionTests.java new file mode 100644 index 000000000000..00a552e942e3 --- /dev/null +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/SanitizingFunctionTests.java @@ -0,0 +1,345 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.actuate.endpoint; + +import java.util.List; +import java.util.Objects; +import java.util.regex.Pattern; + +import org.assertj.core.api.Condition; +import org.assertj.core.api.ObjectAssert; +import org.junit.jupiter.api.Test; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link SanitizingFunction}. + * + * @author Phillip Webb + */ +class SanitizingFunctionTests { + + private static final SanitizableData data = data("key"); + + @Test + void applyUnlessFilteredWhenHasNoFilterReturnsFiltered() { + SanitizingFunction function = SanitizingFunction.sanitizeValue(); + assertThat(function.apply(data)).has(sanitizedValue()); + assertThat(function.applyUnlessFiltered(data)).has(sanitizedValue()); + } + + @Test + void applyUnlessFilteredWhenHasFilterTestingTrueReturnsFiltered() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifMatches((data) -> true); + assertThat(function.apply(data)).has(sanitizedValue()); + assertThat(function.applyUnlessFiltered(data)).has(sanitizedValue()); + } + + @Test + void applyUnlessFilteredWhenHasFilterTestingFalseReturnsUnfiltered() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifMatches((data) -> false); + assertThat(function.apply(data)).has(sanitizedValue()); + assertThat(function.applyUnlessFiltered(data)).has(unsanitizedValue()); + } + + @Test + void ifLikelySensitiveFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifLikelySensitive(); + assertThat(function).satisfies(this::likelyCredentialChecks, this::likelyUriChecks, + this::likelySensitivePropertyChecks, this::vcapServicesChecks); + } + + @Test + void ifLikelyCredentialFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifLikelyCredential(); + assertThat(function).satisfies(this::likelyCredentialChecks); + } + + private void likelyCredentialChecks(SanitizingFunction function) { + assertThatApplyingToKey(function, "password").has(sanitizedValue()); + assertThatApplyingToKey(function, "database.password").has(sanitizedValue()); + assertThatApplyingToKey(function, "PASSWORD").has(sanitizedValue()); + assertThatApplyingToKey(function, "secret").has(sanitizedValue()); + assertThatApplyingToKey(function, "key").has(sanitizedValue()); + assertThatApplyingToKey(function, "token").has(sanitizedValue()); + assertThatApplyingToKey(function, "credentials").has(sanitizedValue()); + assertThatApplyingToKey(function, "thecredentialssecret").has(sanitizedValue()); + assertThatApplyingToKey(function, "some.credentials.here").has(sanitizedValue()); + assertThatApplyingToKey(function, "test").has(unsanitizedValue()); + } + + @Test + void ifLikelyUriFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifLikelyUri(); + assertThat(function).satisfies(this::likelyUriChecks); + } + + private void likelyUriChecks(SanitizingFunction function) { + assertThatApplyingToKey(function, "uri").has(sanitizedValue()); + assertThatApplyingToKey(function, "URI").has(sanitizedValue()); + assertThatApplyingToKey(function, "database.uri").has(sanitizedValue()); + assertThatApplyingToKey(function, "uris").has(sanitizedValue()); + assertThatApplyingToKey(function, "url").has(sanitizedValue()); + assertThatApplyingToKey(function, "urls").has(sanitizedValue()); + assertThatApplyingToKey(function, "address").has(sanitizedValue()); + assertThatApplyingToKey(function, "addresses").has(sanitizedValue()); + assertThatApplyingToKey(function, "test").has(unsanitizedValue()); + } + + @Test + void ifLikelySensitivePropertyFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifLikelySensitiveProperty(); + assertThat(function).satisfies(this::likelySensitivePropertyChecks); + } + + private void likelySensitivePropertyChecks(SanitizingFunction function) { + assertThatApplyingToKey(function, "sun.java.command").has(sanitizedValue()); + assertThatApplyingToKey(function, "spring.application.json").has(sanitizedValue()); + assertThatApplyingToKey(function, "SPRING_APPLICATION_JSON").has(sanitizedValue()); + assertThatApplyingToKey(function, "some.other.json").has(unsanitizedValue()); + } + + @Test + void ifVcapServicesFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifVcapServices(); + assertThat(function).satisfies(this::vcapServicesChecks); + } + + private void vcapServicesChecks(SanitizingFunction function) { + assertThatApplyingToKey(function, "vcap_services").has(sanitizedValue()); + assertThatApplyingToKey(function, "vcap.services").has(sanitizedValue()); + assertThatApplyingToKey(function, "vcap.services.whatever").has(sanitizedValue()); + assertThatApplyingToKey(function, "notvcap.services").has(unsanitizedValue()); + } + + @Test + void ifKeyEqualsFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifKeyEquals("spring", "test"); + assertThatApplyingToKey(function, "spring").has(sanitizedValue()); + assertThatApplyingToKey(function, "SPRING").has(sanitizedValue()); + assertThatApplyingToKey(function, "SpRiNg").has(sanitizedValue()); + assertThatApplyingToKey(function, "test").has(sanitizedValue()); + assertThatApplyingToKey(function, "boot").has(unsanitizedValue()); + assertThatApplyingToKey(function, "xspring").has(unsanitizedValue()); + assertThatApplyingToKey(function, "springx").has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifKeyEndsWithFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifKeyEndsWith("boot", "test"); + assertThatApplyingToKey(function, "springboot").has(sanitizedValue()); + assertThatApplyingToKey(function, "SPRINGboot").has(sanitizedValue()); + assertThatApplyingToKey(function, "springBOOT").has(sanitizedValue()); + assertThatApplyingToKey(function, "boot").has(sanitizedValue()); + assertThatApplyingToKey(function, "atest").has(sanitizedValue()); + assertThatApplyingToKey(function, "bootx").has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifKeyContainsFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifKeyContains("oo", "ee"); + assertThatApplyingToKey(function, "oo").has(sanitizedValue()); + assertThatApplyingToKey(function, "OO").has(sanitizedValue()); + assertThatApplyingToKey(function, "bOOt").has(sanitizedValue()); + assertThatApplyingToKey(function, "boot").has(sanitizedValue()); + assertThatApplyingToKey(function, "beet").has(sanitizedValue()); + assertThatApplyingToKey(function, "spring").has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifKeyMatchesIgnoringCaseFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue() + .ifKeyMatchesIgnoringCase((key, value) -> key.startsWith(value) && key.endsWith(value), "x", "y"); + assertThatApplyingToKey(function, "xtestx").has(sanitizedValue()); + assertThatApplyingToKey(function, "XtestX").has(sanitizedValue()); + assertThatApplyingToKey(function, "YY").has(sanitizedValue()); + assertThatApplyingToKey(function, "xy").has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifKeyMatchesWithRegexFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifKeyMatches("^sp.*$", "^bo.*$"); + assertThatApplyingToKey(function, "spring").has(sanitizedValue()); + assertThatApplyingToKey(function, "spin").has(sanitizedValue()); + assertThatApplyingToKey(function, "SPRING").has(sanitizedValue()); + assertThatApplyingToKey(function, "BOOT").has(sanitizedValue()); + assertThatApplyingToKey(function, "xspring").has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifKeyMatchesWithPatternFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifKeyMatches(Pattern.compile("^sp.*$")); + assertThatApplyingToKey(function, "spring").has(sanitizedValue()); + assertThatApplyingToKey(function, "spin").has(sanitizedValue()); + assertThatApplyingToKey(function, "SPRING").has(unsanitizedValue()); + assertThatApplyingToKey(function, "xspring").has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifKeyMatchesWithPredicatesFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue() + .ifKeyMatches(List.of((key) -> key.startsWith("sp"), (key) -> key.startsWith("BO"))); + assertThatApplyingToKey(function, "spring").has(sanitizedValue()); + assertThatApplyingToKey(function, "spin").has(sanitizedValue()); + assertThatApplyingToKey(function, "BO").has(sanitizedValue()); + assertThatApplyingToKey(function, "SPRING").has(unsanitizedValue()); + assertThatApplyingToKey(function, "boot").has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifKeyMatchesWithPredicateFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifKeyMatches((key) -> key.startsWith("sp")); + assertThatApplyingToKey(function, "spring").has(sanitizedValue()); + assertThatApplyingToKey(function, "spin").has(sanitizedValue()); + assertThatApplyingToKey(function, "boot").has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifValueStringMatchesWithRegexesFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue().ifValueStringMatches("^sp.*$", "^bo.*$"); + assertThatApplyingToValue(function, "spring").has(sanitizedValue()); + assertThatApplyingToValue(function, "SPRING").has(sanitizedValue()); + assertThatApplyingToValue(function, "boot").has(sanitizedValue()); + assertThatApplyingToValue(function, "other").has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifValueStringMatchesWithPatternsFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue() + .ifValueStringMatches(Pattern.compile("^sp.*$")); + assertThatApplyingToValue(function, "spring").has(sanitizedValue()); + assertThatApplyingToValue(function, "spin").has(sanitizedValue()); + assertThatApplyingToValue(function, "SPRING").has(unsanitizedValue()); + assertThatApplyingToValue(function, "xspring").has(unsanitizedValue()); + assertThatApplyingToValue(function, null).has(unsanitizedValue()); + } + + @Test + void ifValueStringStringMatchesWithPredicatesFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue() + .ifValueStringMatches(List.of((value) -> value.startsWith("sp"), (value) -> value.startsWith("BO"))); + assertThatApplyingToValue(function, "spring").has(sanitizedValue()); + assertThatApplyingToValue(function, "spin").has(sanitizedValue()); + assertThatApplyingToValue(function, "BO").has(sanitizedValue()); + assertThatApplyingToValue(function, "SPRING").has(unsanitizedValue()); + assertThatApplyingToValue(function, "boot").has(unsanitizedValue()); + assertThatApplyingToValue(function, null).has(unsanitizedValue()); + } + + @Test + void ifValueStringMatchesWithPredicateFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue() + .ifValueStringMatches((value) -> value.startsWith("sp")); + assertThatApplyingToValue(function, "spring").has(sanitizedValue()); + assertThatApplyingToValue(function, "spin").has(sanitizedValue()); + assertThatApplyingToValue(function, "boot").has(unsanitizedValue()); + assertThatApplyingToValue(function, null).has(unsanitizedValue()); + } + + @Test + void ifValueMatchesWithPredicatesFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue() + .ifValueMatches(List.of((value) -> value instanceof String string && string.startsWith("sp"), + (value) -> value instanceof String string && string.startsWith("BO"))); + assertThatApplyingToValue(function, "spring").has(sanitizedValue()); + assertThatApplyingToValue(function, "spin").has(sanitizedValue()); + assertThatApplyingToValue(function, "BO").has(sanitizedValue()); + assertThatApplyingToValue(function, "SPRING").has(unsanitizedValue()); + assertThatApplyingToValue(function, "boot").has(unsanitizedValue()); + assertThatApplyingToValue(function, 123).has(unsanitizedValue()); + assertThatApplyingToValue(function, null).has(unsanitizedValue()); + } + + @Test + void ifValueMatchesWithPredicateFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue() + .ifValueMatches((value) -> value instanceof String string && string.startsWith("sp")); + assertThatApplyingToValue(function, "spring").has(sanitizedValue()); + assertThatApplyingToValue(function, "spin").has(sanitizedValue()); + assertThatApplyingToValue(function, "boot").has(unsanitizedValue()); + assertThatApplyingToValue(function, 123).has(unsanitizedValue()); + assertThatApplyingToKey(function, null).has(unsanitizedValue()); + } + + @Test + void ifMatchesPredicatesFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue() + .ifMatches(List.of((data) -> data.getKey().startsWith("sp") && "boot".equals(data.getValue()), + (data) -> data.getKey().startsWith("sp") && "framework".equals(data.getValue()))); + assertThatApplying(function, data("spring", "boot")).is(sanitizedValue()); + assertThatApplying(function, data("spring", "framework")).is(sanitizedValue()); + assertThatApplying(function, data("spring", "data")).is(unsanitizedValue()); + assertThatApplying(function, data("spring", null)).is(unsanitizedValue()); + } + + @Test + void ifMatchesPredicateFiltersExpected() { + SanitizingFunction function = SanitizingFunction.sanitizeValue() + .ifMatches((data) -> data.getKey().startsWith("sp") && "boot".equals(data.getValue())); + assertThatApplying(function, data("spring", "boot")).is(sanitizedValue()); + assertThatApplying(function, data("spring", "framework")).is(unsanitizedValue()); + assertThatApplying(function, data("spring", "data")).is(unsanitizedValue()); + assertThatApplying(function, data("spring", null)).is(unsanitizedValue()); + } + + @Test + void ofAllowsChainingFromLambda() { + SanitizingFunction function = SanitizingFunction.of((data) -> data.withValue("----")).ifKeyContains("password"); + assertThat(function.applyUnlessFiltered(data("username", "spring")).getValue()).isEqualTo("spring"); + assertThat(function.applyUnlessFiltered(data("password", "boot")).getValue()).isEqualTo("----"); + } + + private ObjectAssert assertThatApplyingToKey(SanitizingFunction function, String key) { + return assertThatApplying(function, data(key)); + } + + private ObjectAssert assertThatApplyingToValue(SanitizingFunction function, Object value) { + return assertThatApplying(function, data("key", value)); + } + + private ObjectAssert assertThatApplying(SanitizingFunction function, SanitizableData data) { + return assertThat(function.applyUnlessFiltered(data)).as("%s:%s", data.getKey(), data.getValue()); + } + + private Condition sanitizedValue() { + return new Condition<>((data) -> Objects.equals(data.getValue(), SanitizableData.SANITIZED_VALUE), + "sanitized value"); + } + + private Condition unsanitizedValue() { + return new Condition<>((data) -> !Objects.equals(data.getValue(), SanitizableData.SANITIZED_VALUE), + "unsanitized value"); + } + + private static SanitizableData data(String key) { + return data(key, "value"); + } + + private static SanitizableData data(String key, Object value) { + return new SanitizableData(null, key, value); + } + +} diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/DiscoveredOperationMethodTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/DiscoveredOperationMethodTests.java index 9c85115b4161..f04764af6ce6 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/DiscoveredOperationMethodTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/DiscoveredOperationMethodTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,7 +42,7 @@ void createWhenAnnotationAttributesIsNullShouldThrowException() { Method method = ReflectionUtils.findMethod(getClass(), "example"); assertThatIllegalArgumentException() .isThrownBy(() -> new DiscoveredOperationMethod(method, OperationType.READ, null)) - .withMessageContaining("AnnotationAttributes must not be null"); + .withMessageContaining("'annotationAttributes' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/DiscovererEndpointFilterTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/DiscovererEndpointFilterTests.java index b457b2fa951d..ecaa0a2f36d5 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/DiscovererEndpointFilterTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/DiscovererEndpointFilterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,7 +43,7 @@ class DiscovererEndpointFilterTests { @Test void createWhenDiscovererIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new TestDiscovererEndpointFilter(null)) - .withMessageContaining("Discoverer must not be null"); + .withMessageContaining("'discoverer' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/EndpointDiscovererTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/EndpointDiscovererTests.java index 131e0e76ef72..2c3dc531f4ec 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/EndpointDiscovererTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/annotation/EndpointDiscovererTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -76,7 +76,7 @@ void createWhenApplicationContextIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TestEndpointDiscoverer(null, mock(ParameterValueMapper.class), Collections.emptyList(), Collections.emptyList())) - .withMessageContaining("ApplicationContext must not be null"); + .withMessageContaining("'applicationContext' must not be null"); } @Test @@ -84,7 +84,7 @@ void createWhenParameterValueMapperIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TestEndpointDiscoverer(mock(ApplicationContext.class), null, Collections.emptyList(), Collections.emptyList())) - .withMessageContaining("ParameterValueMapper must not be null"); + .withMessageContaining("'parameterValueMapper' must not be null"); } @Test @@ -92,7 +92,7 @@ void createWhenInvokerAdvisorsIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TestEndpointDiscoverer(mock(ApplicationContext.class), mock(ParameterValueMapper.class), null, Collections.emptyList())) - .withMessageContaining("InvokerAdvisors must not be null"); + .withMessageContaining("'invokerAdvisors' must not be null"); } @Test @@ -100,7 +100,7 @@ void createWhenFiltersIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TestEndpointDiscoverer(mock(ApplicationContext.class), mock(ParameterValueMapper.class), Collections.emptyList(), null)) - .withMessageContaining("Filters must not be null"); + .withMessageContaining("'endpointFilters' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodParametersTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodParametersTests.java index 7f7a1f12bbec..8c2d7ebf1181 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodParametersTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodParametersTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -51,13 +51,13 @@ class OperationMethodParametersTests { void createWhenMethodIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new OperationMethodParameters(null, mock(ParameterNameDiscoverer.class))) - .withMessageContaining("Method must not be null"); + .withMessageContaining("'method' must not be null"); } @Test void createWhenParameterNameDiscovererIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new OperationMethodParameters(this.exampleMethod, null)) - .withMessageContaining("ParameterNameDiscoverer must not be null"); + .withMessageContaining("'parameterNameDiscoverer' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodTests.java index 46dea6abe312..bc4856fbd251 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/OperationMethodTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,13 +39,13 @@ class OperationMethodTests { @Test void createWhenMethodIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new OperationMethod(null, OperationType.READ)) - .withMessageContaining("Method must not be null"); + .withMessageContaining("'method' must not be null"); } @Test void createWhenOperationTypeIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new OperationMethod(this.exampleMethod, null)) - .withMessageContaining("OperationType must not be null"); + .withMessageContaining("'operationType' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/ReflectiveOperationInvokerTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/ReflectiveOperationInvokerTests.java index 9e7cab081a6d..a59cbabd6b2e 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/ReflectiveOperationInvokerTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoke/reflect/ReflectiveOperationInvokerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -60,21 +60,21 @@ void setup() { void createWhenTargetIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new ReflectiveOperationInvoker(null, this.operationMethod, this.parameterValueMapper)) - .withMessageContaining("Target must not be null"); + .withMessageContaining("'target' must not be null"); } @Test void createWhenOperationMethodIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new ReflectiveOperationInvoker(this.target, null, this.parameterValueMapper)) - .withMessageContaining("OperationMethod must not be null"); + .withMessageContaining("'operationMethod' must not be null"); } @Test void createWhenParameterValueMapperIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new ReflectiveOperationInvoker(this.target, this.operationMethod, null)) - .withMessageContaining("ParameterValueMapper must not be null"); + .withMessageContaining("'parameterValueMapper' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoker/cache/CachingOperationInvokerTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoker/cache/CachingOperationInvokerTests.java index 33f89600192d..477c9e07942a 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoker/cache/CachingOperationInvokerTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/invoker/cache/CachingOperationInvokerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -58,7 +58,7 @@ class CachingOperationInvokerTests { void createInstanceWithTtlSetToZero() { assertThatIllegalArgumentException() .isThrownBy(() -> new CachingOperationInvoker(mock(OperationInvoker.class), 0)) - .withMessageContaining("TimeToLive"); + .withMessage("'timeToLive' must be greater than zero"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/jmx/EndpointMBeanTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/jmx/EndpointMBeanTests.java index 442bd367c948..6a8d59f4b470 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/jmx/EndpointMBeanTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/jmx/EndpointMBeanTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -63,14 +63,14 @@ class EndpointMBeanTests { void createWhenResponseMapperIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new EndpointMBean(null, null, mock(ExposableJmxEndpoint.class))) - .withMessageContaining("ResponseMapper must not be null"); + .withMessageContaining("'responseMapper' must not be null"); } @Test void createWhenEndpointIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new EndpointMBean(mock(JmxOperationResponseMapper.class), null, null)) - .withMessageContaining("Endpoint must not be null"); + .withMessageContaining("'endpoint' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/jmx/JmxEndpointExporterTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/jmx/JmxEndpointExporterTests.java index cfd0a8d21622..641529a2f120 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/jmx/JmxEndpointExporterTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/jmx/JmxEndpointExporterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -77,21 +77,21 @@ void createWhenMBeanServerIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy( () -> new JmxEndpointExporter(null, this.objectNameFactory, this.responseMapper, this.endpoints)) - .withMessageContaining("MBeanServer must not be null"); + .withMessageContaining("'mBeanServer' must not be null"); } @Test void createWhenObjectNameFactoryIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new JmxEndpointExporter(this.mBeanServer, null, this.responseMapper, this.endpoints)) - .withMessageContaining("ObjectNameFactory must not be null"); + .withMessageContaining("'objectNameFactory' must not be null"); } @Test void createWhenResponseMapperIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new JmxEndpointExporter(this.mBeanServer, this.objectNameFactory, null, this.endpoints)) - .withMessageContaining("ResponseMapper must not be null"); + .withMessageContaining("'responseMapper' must not be null"); } @Test @@ -99,7 +99,7 @@ void createWhenEndpointsIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy( () -> new JmxEndpointExporter(this.mBeanServer, this.objectNameFactory, this.responseMapper, null)) - .withMessageContaining("Endpoints must not be null"); + .withMessageContaining("'endpoints' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/EndpointMediaTypesTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/EndpointMediaTypesTests.java index a9495d188e51..ae672f956869 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/EndpointMediaTypesTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/EndpointMediaTypesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,13 +47,13 @@ void defaultReturnsExpectedProducedAndConsumedTypes() { @Test void createWhenProducedIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new EndpointMediaTypes(null, Collections.emptyList())) - .withMessageContaining("Produced must not be null"); + .withMessageContaining("'produced' must not be null"); } @Test void createWhenConsumedIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new EndpointMediaTypes(Collections.emptyList(), null)) - .withMessageContaining("Consumed must not be null"); + .withMessageContaining("'consumed' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/EndpointServletTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/EndpointServletTests.java index 1a1d8cf247bb..91eaec87ac7f 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/EndpointServletTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/EndpointServletTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,13 +42,13 @@ class EndpointServletTests { @Test void createWhenServletClassIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new EndpointServlet((Class) null)) - .withMessageContaining("Servlet must not be null"); + .withMessageContaining("'servlet' must not be null"); } @Test void createWhenServletIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new EndpointServlet((Servlet) null)) - .withMessageContaining("Servlet must not be null"); + .withMessageContaining("'servlet' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/LinkTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/LinkTests.java index 65f5a43183ca..104e56bb1283 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/LinkTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/LinkTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ class LinkTests { @Test void createWhenHrefIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new Link(null)) - .withMessageContaining("HREF must not be null"); + .withMessageContaining("'href' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/PathMappedEndpointsTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/PathMappedEndpointsTests.java index 1d5b2b466c15..38d4865b1d20 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/PathMappedEndpointsTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/PathMappedEndpointsTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,14 +43,14 @@ class PathMappedEndpointsTests { void createWhenSupplierIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new PathMappedEndpoints(null, (WebEndpointsSupplier) null)) - .withMessageContaining("Supplier must not be null"); + .withMessageContaining("'supplier' must not be null"); } @Test void createWhenSuppliersIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new PathMappedEndpoints(null, (Collection>) null)) - .withMessageContaining("Suppliers must not be null"); + .withMessageContaining("'suppliers' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/ServletEndpointRegistrarTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/ServletEndpointRegistrarTests.java index 6e9c9dacae1b..2c698bcdda82 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/ServletEndpointRegistrarTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/ServletEndpointRegistrarTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -68,7 +68,7 @@ class ServletEndpointRegistrarTests { @Test void createWhenServletEndpointsIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new ServletEndpointRegistrar(null, null)) - .withMessageContaining("ServletEndpoints must not be null"); + .withMessageContaining("'servletEndpoints' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/reactive/WebFluxEndpointHandlerMappingTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/reactive/WebFluxEndpointHandlerMappingTests.java index 988853cc210e..b7dd80757911 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/reactive/WebFluxEndpointHandlerMappingTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/reactive/WebFluxEndpointHandlerMappingTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ class WebFluxEndpointHandlerMappingTests { void shouldRegisterHints() { RuntimeHints runtimeHints = new RuntimeHints(); new WebFluxEndpointHandlerMappingRuntimeHints().registerHints(runtimeHints, getClass().getClassLoader()); - assertThat(RuntimeHintsPredicates.reflection().onMethod(WebFluxLinksHandler.class, "links")) + assertThat(RuntimeHintsPredicates.reflection().onMethod(WebFluxLinksHandler.class, "links").invoke()) .accepts(runtimeHints); assertThat(RuntimeHintsPredicates.reflection().onType(Link.class)).accepts(runtimeHints); } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/servlet/WebMvcEndpointHandlerMappingTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/servlet/WebMvcEndpointHandlerMappingTests.java index 11bd7ee410fa..9a8d9d613552 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/servlet/WebMvcEndpointHandlerMappingTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/endpoint/web/servlet/WebMvcEndpointHandlerMappingTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ class WebMvcEndpointHandlerMappingTests { void shouldRegisterHints() { RuntimeHints runtimeHints = new RuntimeHints(); new WebMvcEndpointHandlerMappingRuntimeHints().registerHints(runtimeHints, getClass().getClassLoader()); - assertThat(RuntimeHintsPredicates.reflection().onMethod(WebMvcLinksHandler.class, "links")) + assertThat(RuntimeHintsPredicates.reflection().onMethod(WebMvcLinksHandler.class, "links").invoke()) .accepts(runtimeHints); assertThat(RuntimeHintsPredicates.reflection().onType(Link.class)).accepts(runtimeHints); } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/CompositeHealthContributorReactiveAdapterTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/CompositeHealthContributorReactiveAdapterTests.java index 1a2916ad08b2..7b37f84eec76 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/CompositeHealthContributorReactiveAdapterTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/CompositeHealthContributorReactiveAdapterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ class CompositeHealthContributorReactiveAdapterTests { @Test void createWhenDelegateIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new CompositeHealthContributorReactiveAdapter(null)) - .withMessage("Delegate must not be null"); + .withMessage("'delegate' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/CompositeHealthTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/CompositeHealthTests.java index 0d8d51112eee..57b70b3ace28 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/CompositeHealthTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/CompositeHealthTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,7 @@ class CompositeHealthTests { void createWhenStatusIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new CompositeHealth(ApiVersion.V3, null, Collections.emptyMap())) - .withMessage("Status must not be null"); + .withMessage("'status' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthEndpointGroupsTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthEndpointGroupsTests.java index a9e8058a563f..6f16226f1e01 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthEndpointGroupsTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthEndpointGroupsTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,14 +34,14 @@ class HealthEndpointGroupsTests { @Test void ofWhenPrimaryIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> HealthEndpointGroups.of(null, Collections.emptyMap())) - .withMessage("Primary must not be null"); + .withMessage("'primary' must not be null"); } @Test void ofWhenAdditionalIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> HealthEndpointGroups.of(mock(HealthEndpointGroup.class), null)) - .withMessage("Additional must not be null"); + .withMessage("'additional' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthEndpointSupportTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthEndpointSupportTests.java index 8af31ccdbf44..59e08c0430e9 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthEndpointSupportTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthEndpointSupportTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -64,13 +64,13 @@ abstract class HealthEndpointSupportTests, @Test void createWhenRegistryIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> create(null, this.groups)) - .withMessage("Registry must not be null"); + .withMessage("'registry' must not be null"); } @Test void createWhenGroupsIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> create(this.registry, null)) - .withMessage("Groups must not be null"); + .withMessage("'groups' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthTests.java index 5f0aba842170..990b0493af6d 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/HealthTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,7 +40,7 @@ class HealthTests { @Test void statusMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new Health.Builder(null, null)) - .withMessageContaining("Status must not be null"); + .withMessageContaining("'status' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/NamedContributorTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/NamedContributorTests.java index a26c1e460e55..abab9106ab78 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/NamedContributorTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/NamedContributorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,13 +38,13 @@ void ofNameAndContributorCreatesContributor() { @Test void ofWhenNameIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> NamedContributor.of(null, "two")) - .withMessage("Name must not be null"); + .withMessage("'name' must not be null"); } @Test void ofWhenContributorIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> NamedContributor.of("one", null)) - .withMessage("Contributor must not be null"); + .withMessage("'contributor' must not be null"); } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/NamedContributorsMapAdapterTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/NamedContributorsMapAdapterTests.java index 9794a32b09cf..b673d85c5698 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/NamedContributorsMapAdapterTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/NamedContributorsMapAdapterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,14 +40,14 @@ class NamedContributorsMapAdapterTests { void createWhenMapIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TestNamedContributorsMapAdapter<>(null, Function.identity())) - .withMessage("Map must not be null"); + .withMessage("'map' must not be null"); } @Test void createWhenValueAdapterIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TestNamedContributorsMapAdapter<>(Collections.emptyMap(), null)) - .withMessage("ValueAdapter must not be null"); + .withMessage("'valueAdapter' must not be null"); } @Test @@ -55,7 +55,7 @@ void createWhenMapContainsNullValueThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TestNamedContributorsMapAdapter<>(Collections.singletonMap("test", null), Function.identity())) - .withMessage("Map must not contain null values"); + .withMessage("'map' must not contain null values"); } @Test @@ -63,7 +63,7 @@ void createWhenMapContainsNullKeyThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TestNamedContributorsMapAdapter<>(Collections.singletonMap(null, "test"), Function.identity())) - .withMessage("Map must not contain null keys"); + .withMessage("'map' must not contain null keys"); } @Test @@ -71,7 +71,7 @@ void createWhenMapContainsKeyWithSlashThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TestNamedContributorsMapAdapter<>(Collections.singletonMap("test/key", "test"), Function.identity())) - .withMessage("Map keys must not contain a '/'"); + .withMessage("'map' keys must not contain a '/'"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/ReactiveHealthContributorTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/ReactiveHealthContributorTests.java index 2acf5a61f3ca..545c5217c6c1 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/ReactiveHealthContributorTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/ReactiveHealthContributorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ class ReactiveHealthContributorTests { @Test void adaptWhenNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ReactiveHealthContributor.adapt(null)) - .withMessage("HealthContributor must not be null"); + .withMessage("'healthContributor' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/StatusTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/StatusTests.java index f9284dd4331d..91f27a99643c 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/StatusTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/health/StatusTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,13 +32,13 @@ class StatusTests { @Test void createWhenCodeIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new Status(null, "")) - .withMessage("Code must not be null"); + .withMessage("'code' must not be null"); } @Test void createWhenDescriptionIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new Status("code", null)) - .withMessage("Description must not be null"); + .withMessage("'description' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/info/ProcessInfoContributorTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/info/ProcessInfoContributorTests.java index faceb15528ed..e437802566ba 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/info/ProcessInfoContributorTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/info/ProcessInfoContributorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,9 +17,12 @@ package org.springframework.boot.actuate.info; import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.condition.EnabledForJreRange; +import org.junit.jupiter.api.condition.JRE; import org.springframework.aot.hint.MemberCategory; import org.springframework.aot.hint.RuntimeHints; +import org.springframework.aot.hint.TypeReference; import org.springframework.aot.hint.predicate.RuntimeHintsPredicates; import org.springframework.boot.actuate.info.ProcessInfoContributor.ProcessInfoContributorRuntimeHints; import org.springframework.boot.info.ProcessInfo; @@ -30,6 +33,7 @@ * Tests for {@link ProcessInfoContributor}. * * @author Jonatan Ivanov + * @author Moritz Halbritter */ class ProcessInfoContributorTests { @@ -52,4 +56,14 @@ void shouldRegisterHints() { .accepts(runtimeHints); } + @Test + @EnabledForJreRange(min = JRE.JAVA_24) + void shouldRegisterRuntimeHintsForVirtualThreadSchedulerMXBean() { + RuntimeHints runtimeHints = new RuntimeHints(); + new ProcessInfoContributorRuntimeHints().registerHints(runtimeHints, getClass().getClassLoader()); + assertThat(RuntimeHintsPredicates.reflection() + .onType(TypeReference.of("jdk.management.VirtualThreadSchedulerMXBean")) + .withMemberCategories(MemberCategory.INVOKE_PUBLIC_METHODS)).accepts(runtimeHints); + } + } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/logging/LoggersEndpointTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/logging/LoggersEndpointTests.java index 887fa339d315..1bd09a381bed 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/logging/LoggersEndpointTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/logging/LoggersEndpointTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -164,13 +164,17 @@ void registersRuntimeHintsForClassesSerializedToJson() { new ReflectiveRuntimeHintsRegistrar().registerRuntimeHints(runtimeHints, LoggersEndpoint.class); ReflectionHintsPredicates reflection = RuntimeHintsPredicates.reflection(); assertThat(reflection.onType(LoggerLevelsDescriptor.class)).accepts(runtimeHints); - assertThat(reflection.onMethod(LoggerLevelsDescriptor.class, "getConfiguredLevel")).accepts(runtimeHints); + assertThat(reflection.onMethod(LoggerLevelsDescriptor.class, "getConfiguredLevel").invoke()) + .accepts(runtimeHints); assertThat(reflection.onType(SingleLoggerLevelsDescriptor.class)).accepts(runtimeHints); - assertThat(reflection.onMethod(SingleLoggerLevelsDescriptor.class, "getEffectiveLevel")).accepts(runtimeHints); - assertThat(reflection.onMethod(SingleLoggerLevelsDescriptor.class, "getConfiguredLevel")).accepts(runtimeHints); + assertThat(reflection.onMethod(SingleLoggerLevelsDescriptor.class, "getEffectiveLevel").invoke()) + .accepts(runtimeHints); + assertThat(reflection.onMethod(SingleLoggerLevelsDescriptor.class, "getConfiguredLevel").invoke()) + .accepts(runtimeHints); assertThat(reflection.onType(GroupLoggerLevelsDescriptor.class)).accepts(runtimeHints); - assertThat(reflection.onMethod(GroupLoggerLevelsDescriptor.class, "getMembers")).accepts(runtimeHints); - assertThat(reflection.onMethod(GroupLoggerLevelsDescriptor.class, "getConfiguredLevel")).accepts(runtimeHints); + assertThat(reflection.onMethod(GroupLoggerLevelsDescriptor.class, "getMembers").invoke()).accepts(runtimeHints); + assertThat(reflection.onMethod(GroupLoggerLevelsDescriptor.class, "getConfiguredLevel").invoke()) + .accepts(runtimeHints); } } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/cache/HazelcastCacheMeterBinderProviderTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/cache/HazelcastCacheMeterBinderProviderTests.java index a38b49b47cd1..d3699fcc69a6 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/cache/HazelcastCacheMeterBinderProviderTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/cache/HazelcastCacheMeterBinderProviderTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -55,7 +55,7 @@ void hazelcastCacheProvider() { void shouldRegisterHints() { RuntimeHints runtimeHints = new RuntimeHints(); new HazelcastCacheMeterBinderProviderRuntimeHints().registerHints(runtimeHints, getClass().getClassLoader()); - assertThat(RuntimeHintsPredicates.reflection().onMethod(HazelcastCache.class, "getNativeCache")) + assertThat(RuntimeHintsPredicates.reflection().onMethod(HazelcastCache.class, "getNativeCache").invoke()) .accepts(runtimeHints); assertThat(RuntimeHintsPredicates.reflection().onType(HazelcastCacheMetrics.class)).accepts(runtimeHints); } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusPushGatewayManagerTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusPushGatewayManagerTests.java index 6960ea722752..cc14ef1c1670 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusPushGatewayManagerTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusPushGatewayManagerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,12 +17,9 @@ package org.springframework.boot.actuate.metrics.export.prometheus; import java.time.Duration; -import java.util.Collections; -import java.util.Map; import java.util.concurrent.ScheduledFuture; -import io.prometheus.client.CollectorRegistry; -import io.prometheus.client.exporter.PushGateway; +import io.prometheus.metrics.exporter.pushgateway.PushGateway; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.ExtendWith; import org.mockito.ArgumentCaptor; @@ -55,16 +52,11 @@ class PrometheusPushGatewayManagerTests { @Mock private PushGateway pushGateway; - @Mock - private CollectorRegistry registry; - @Mock private TaskScheduler scheduler; private final Duration pushRate = Duration.ofSeconds(1); - private final Map groupingKey = Collections.singletonMap("foo", "bar"); - @Captor private ArgumentCaptor task; @@ -74,68 +66,48 @@ class PrometheusPushGatewayManagerTests { @Test void createWhenPushGatewayIsNullThrowsException() { assertThatIllegalArgumentException() - .isThrownBy(() -> new PrometheusPushGatewayManager(null, this.registry, this.scheduler, this.pushRate, - "job", this.groupingKey, null)) - .withMessage("PushGateway must not be null"); - } - - @Test - void createWhenCollectorRegistryIsNullThrowsException() { - assertThatIllegalArgumentException() - .isThrownBy(() -> new PrometheusPushGatewayManager(this.pushGateway, null, this.scheduler, this.pushRate, - "job", this.groupingKey, null)) - .withMessage("Registry must not be null"); + .isThrownBy(() -> new PrometheusPushGatewayManager(null, this.scheduler, this.pushRate, null)) + .withMessage("'pushGateway' must not be null"); } @Test void createWhenSchedulerIsNullThrowsException() { assertThatIllegalArgumentException() - .isThrownBy(() -> new PrometheusPushGatewayManager(this.pushGateway, this.registry, null, this.pushRate, - "job", this.groupingKey, null)) - .withMessage("Scheduler must not be null"); + .isThrownBy(() -> new PrometheusPushGatewayManager(this.pushGateway, null, this.pushRate, null)) + .withMessage("'scheduler' must not be null"); } @Test void createWhenPushRateIsNullThrowsException() { assertThatIllegalArgumentException() - .isThrownBy(() -> new PrometheusPushGatewayManager(this.pushGateway, this.registry, this.scheduler, null, - "job", this.groupingKey, null)) - .withMessage("PushRate must not be null"); - } - - @Test - void createWhenJobIsEmptyThrowsException() { - assertThatIllegalArgumentException() - .isThrownBy(() -> new PrometheusPushGatewayManager(this.pushGateway, this.registry, this.scheduler, - this.pushRate, "", this.groupingKey, null)) - .withMessage("Job must not be empty"); + .isThrownBy(() -> new PrometheusPushGatewayManager(this.pushGateway, this.scheduler, null, null)) + .withMessage("'pushRate' must not be null"); } @Test void createShouldSchedulePushAsFixedRate() throws Exception { - new PrometheusPushGatewayManager(this.pushGateway, this.registry, this.scheduler, this.pushRate, "job", - this.groupingKey, null); + new PrometheusPushGatewayManager(this.pushGateway, this.scheduler, this.pushRate, null); then(this.scheduler).should().scheduleAtFixedRate(this.task.capture(), eq(this.pushRate)); this.task.getValue().run(); - then(this.pushGateway).should().pushAdd(this.registry, "job", this.groupingKey); + then(this.pushGateway).should().pushAdd(); } @Test - void shutdownWhenOwnsSchedulerDoesShutdownScheduler() { + void shutdownWhenOwnsSchedulerDoesShutDownScheduler() { PushGatewayTaskScheduler ownedScheduler = givenScheduleAtFixedRateWillReturnFuture( mock(PushGatewayTaskScheduler.class)); - PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.registry, - ownedScheduler, this.pushRate, "job", this.groupingKey, null); + PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, ownedScheduler, + this.pushRate, null); manager.shutdown(); then(ownedScheduler).should().shutdown(); } @Test - void shutdownWhenDoesNotOwnSchedulerDoesNotShutdownScheduler() { + void shutdownWhenDoesNotOwnSchedulerDoesNotShutDownScheduler() { ThreadPoolTaskScheduler otherScheduler = givenScheduleAtFixedRateWillReturnFuture( mock(ThreadPoolTaskScheduler.class)); - PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.registry, - otherScheduler, this.pushRate, "job", this.groupingKey, null); + PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, otherScheduler, + this.pushRate, null); manager.shutdown(); then(otherScheduler).should(never()).shutdown(); } @@ -143,38 +115,38 @@ void shutdownWhenDoesNotOwnSchedulerDoesNotShutdownScheduler() { @Test void shutdownWhenShutdownOperationIsPostPerformsPushAddOnShutdown() throws Exception { givenScheduleAtFixedRateWithReturnFuture(); - PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.registry, - this.scheduler, this.pushRate, "job", this.groupingKey, ShutdownOperation.POST); + PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.scheduler, + this.pushRate, ShutdownOperation.POST); manager.shutdown(); then(this.future).should().cancel(false); - then(this.pushGateway).should().pushAdd(this.registry, "job", this.groupingKey); + then(this.pushGateway).should().pushAdd(); } @Test void shutdownWhenShutdownOperationIsPutPerformsPushOnShutdown() throws Exception { givenScheduleAtFixedRateWithReturnFuture(); - PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.registry, - this.scheduler, this.pushRate, "job", this.groupingKey, ShutdownOperation.PUT); + PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.scheduler, + this.pushRate, ShutdownOperation.PUT); manager.shutdown(); then(this.future).should().cancel(false); - then(this.pushGateway).should().push(this.registry, "job", this.groupingKey); + then(this.pushGateway).should().push(); } @Test void shutdownWhenShutdownOperationIsDeletePerformsDeleteOnShutdown() throws Exception { givenScheduleAtFixedRateWithReturnFuture(); - PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.registry, - this.scheduler, this.pushRate, "job", this.groupingKey, ShutdownOperation.DELETE); + PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.scheduler, + this.pushRate, ShutdownOperation.DELETE); manager.shutdown(); then(this.future).should().cancel(false); - then(this.pushGateway).should().delete("job", this.groupingKey); + then(this.pushGateway).should().delete(); } @Test void shutdownWhenShutdownOperationIsNoneDoesNothing() { givenScheduleAtFixedRateWithReturnFuture(); - PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.registry, - this.scheduler, this.pushRate, "job", this.groupingKey, ShutdownOperation.NONE); + PrometheusPushGatewayManager manager = new PrometheusPushGatewayManager(this.pushGateway, this.scheduler, + this.pushRate, ShutdownOperation.NONE); manager.shutdown(); then(this.future).should().cancel(false); then(this.pushGateway).shouldHaveNoInteractions(); @@ -182,10 +154,9 @@ void shutdownWhenShutdownOperationIsNoneDoesNothing() { @Test void pushDoesNotThrowException() throws Exception { - new PrometheusPushGatewayManager(this.pushGateway, this.registry, this.scheduler, this.pushRate, "job", - this.groupingKey, null); + new PrometheusPushGatewayManager(this.pushGateway, this.scheduler, this.pushRate, null); then(this.scheduler).should().scheduleAtFixedRate(this.task.capture(), eq(this.pushRate)); - willThrow(RuntimeException.class).given(this.pushGateway).pushAdd(this.registry, "job", this.groupingKey); + willThrow(RuntimeException.class).given(this.pushGateway).pushAdd(); this.task.getValue().run(); } diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusSimpleclientScrapeEndpointIntegrationTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusSimpleclientScrapeEndpointIntegrationTests.java deleted file mode 100644 index 2b2a0aa328c1..000000000000 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/PrometheusSimpleclientScrapeEndpointIntegrationTests.java +++ /dev/null @@ -1,150 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.metrics.export.prometheus; - -import io.micrometer.core.instrument.Clock; -import io.micrometer.core.instrument.Counter; -import io.micrometer.core.instrument.MeterRegistry; -import io.prometheus.client.CollectorRegistry; -import io.prometheus.client.exporter.common.TextFormat; - -import org.springframework.boot.actuate.endpoint.web.test.WebEndpointTest; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.http.MediaType; -import org.springframework.test.web.reactive.server.WebTestClient; - -import static org.assertj.core.api.Assertions.assertThat; - -/** - * Tests for {@link PrometheusSimpleclientScrapeEndpoint}. - * - * @author Jon Schneider - * @author Johnny Lim - */ -@SuppressWarnings({ "deprecation", "removal" }) -class PrometheusSimpleclientScrapeEndpointIntegrationTests { - - @WebEndpointTest - void scrapeHasContentTypeText004ByDefault(WebTestClient client) { - String expectedContentType = TextFormat.CONTENT_TYPE_004; - assertThat(TextFormat.chooseContentType(null)).isEqualTo(expectedContentType); - client.get() - .uri("/actuator/prometheus") - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(MediaType.parseMediaType(expectedContentType)) - .expectBody(String.class) - .value((body) -> assertThat(body).contains("counter1_total") - .contains("counter2_total") - .contains("counter3_total")); - } - - @WebEndpointTest - void scrapeHasContentTypeText004ByDefaultWhenClientAcceptsWildcardWithParameter(WebTestClient client) { - String expectedContentType = TextFormat.CONTENT_TYPE_004; - String accept = "*/*;q=0.8"; - assertThat(TextFormat.chooseContentType(accept)).isEqualTo(expectedContentType); - client.get() - .uri("/actuator/prometheus") - .accept(MediaType.parseMediaType(accept)) - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(MediaType.parseMediaType(expectedContentType)) - .expectBody(String.class) - .value((body) -> assertThat(body).contains("counter1_total") - .contains("counter2_total") - .contains("counter3_total")); - } - - @WebEndpointTest - void scrapeCanProduceOpenMetrics100(WebTestClient client) { - MediaType openMetrics = MediaType.parseMediaType(TextFormat.CONTENT_TYPE_OPENMETRICS_100); - client.get() - .uri("/actuator/prometheus") - .accept(openMetrics) - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(openMetrics) - .expectBody(String.class) - .value((body) -> assertThat(body).contains("counter1_total") - .contains("counter2_total") - .contains("counter3_total")); - } - - @WebEndpointTest - void scrapePrefersToProduceOpenMetrics100(WebTestClient client) { - MediaType openMetrics = MediaType.parseMediaType(TextFormat.CONTENT_TYPE_OPENMETRICS_100); - MediaType textPlain = MediaType.parseMediaType(TextFormat.CONTENT_TYPE_004); - client.get() - .uri("/actuator/prometheus") - .accept(openMetrics, textPlain) - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(openMetrics); - } - - @WebEndpointTest - void scrapeWithIncludedNames(WebTestClient client) { - client.get() - .uri("/actuator/prometheus?includedNames=counter1_total,counter2_total") - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(MediaType.parseMediaType(TextFormat.CONTENT_TYPE_004)) - .expectBody(String.class) - .value((body) -> assertThat(body).contains("counter1_total") - .contains("counter2_total") - .doesNotContain("counter3_total")); - } - - @Configuration(proxyBeanMethods = false) - static class TestConfiguration { - - @Bean - PrometheusSimpleclientScrapeEndpoint prometheusScrapeEndpoint(CollectorRegistry collectorRegistry) { - return new PrometheusSimpleclientScrapeEndpoint(collectorRegistry); - } - - @Bean - CollectorRegistry collectorRegistry() { - return new CollectorRegistry(true); - } - - @Bean - @SuppressWarnings("deprecation") - MeterRegistry registry(CollectorRegistry registry) { - io.micrometer.prometheus.PrometheusMeterRegistry meterRegistry = new io.micrometer.prometheus.PrometheusMeterRegistry( - (k) -> null, registry, Clock.SYSTEM); - Counter.builder("counter1").register(meterRegistry); - Counter.builder("counter2").register(meterRegistry); - Counter.builder("counter3").register(meterRegistry); - return meterRegistry; - } - - } - -} diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/SecondCustomPrometheusScrapeEndpointIntegrationTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/SecondCustomPrometheusScrapeEndpointIntegrationTests.java deleted file mode 100644 index 2f47b4378676..000000000000 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/metrics/export/prometheus/SecondCustomPrometheusScrapeEndpointIntegrationTests.java +++ /dev/null @@ -1,210 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.actuate.metrics.export.prometheus; - -import java.util.Properties; - -import io.micrometer.core.instrument.Clock; -import io.micrometer.core.instrument.Counter; -import io.micrometer.core.instrument.composite.CompositeMeterRegistry; -import io.micrometer.prometheusmetrics.PrometheusMeterRegistry; -import io.prometheus.client.CollectorRegistry; -import io.prometheus.metrics.expositionformats.OpenMetricsTextFormatWriter; -import io.prometheus.metrics.expositionformats.PrometheusTextFormatWriter; -import io.prometheus.metrics.model.registry.PrometheusRegistry; - -import org.springframework.boot.actuate.endpoint.web.annotation.WebEndpoint; -import org.springframework.boot.actuate.endpoint.web.test.WebEndpointTest; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.http.MediaType; -import org.springframework.test.web.reactive.server.WebTestClient; - -import static org.assertj.core.api.Assertions.assertThat; - -/** - * Tests for exposing a {@link PrometheusScrapeEndpoint} and - * {@link PrometheusSimpleclientScrapeEndpoint} with different IDs. - * - * @author Jon Schneider - * @author Johnny Lim - */ -class SecondCustomPrometheusScrapeEndpointIntegrationTests { - - @WebEndpointTest - void scrapeHasContentTypeText004ByDefault(WebTestClient client) { - scrapeHasContentTypeText004ByDefault(client, "/actuator/prometheus"); - scrapeHasContentTypeText004ByDefault(client, "/actuator/prometheussc"); - } - - private void scrapeHasContentTypeText004ByDefault(WebTestClient client, String uri) { - String expectedContentType = PrometheusTextFormatWriter.CONTENT_TYPE; - client.get() - .uri(uri) - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(MediaType.parseMediaType(expectedContentType)) - .expectBody(String.class) - .value((body) -> assertThat(body).contains("counter1_total") - .contains("counter2_total") - .contains("counter3_total")); - } - - @WebEndpointTest - void scrapeHasContentTypeText004ByDefaultWhenClientAcceptsWildcardWithParameter(WebTestClient client) { - scrapeHasContentTypeText004ByDefaultWhenClientAcceptsWildcardWithParameter(client, "/actuator/prometheus"); - scrapeHasContentTypeText004ByDefaultWhenClientAcceptsWildcardWithParameter(client, "/actuator/prometheussc"); - } - - private void scrapeHasContentTypeText004ByDefaultWhenClientAcceptsWildcardWithParameter(WebTestClient client, - String uri) { - String expectedContentType = PrometheusTextFormatWriter.CONTENT_TYPE; - String accept = "*/*;q=0.8"; - client.get() - .uri(uri) - .accept(MediaType.parseMediaType(accept)) - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(MediaType.parseMediaType(expectedContentType)) - .expectBody(String.class) - .value((body) -> assertThat(body).contains("counter1_total") - .contains("counter2_total") - .contains("counter3_total")); - } - - @WebEndpointTest - void scrapeCanProduceOpenMetrics100(WebTestClient client) { - scrapeCanProduceOpenMetrics100(client, "/actuator/prometheus"); - scrapeCanProduceOpenMetrics100(client, "/actuator/prometheussc"); - } - - private void scrapeCanProduceOpenMetrics100(WebTestClient client, String uri) { - MediaType openMetrics = MediaType.parseMediaType(OpenMetricsTextFormatWriter.CONTENT_TYPE); - client.get() - .uri(uri) - .accept(openMetrics) - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(openMetrics) - .expectBody(String.class) - .value((body) -> assertThat(body).contains("counter1_total") - .contains("counter2_total") - .contains("counter3_total")); - } - - @WebEndpointTest - void scrapePrefersToProduceOpenMetrics100(WebTestClient client) { - scrapePrefersToProduceOpenMetrics100(client, "/actuator/prometheus"); - scrapePrefersToProduceOpenMetrics100(client, "/actuator/prometheussc"); - } - - private void scrapePrefersToProduceOpenMetrics100(WebTestClient client, String uri) { - MediaType openMetrics = MediaType.parseMediaType(OpenMetricsTextFormatWriter.CONTENT_TYPE); - MediaType textPlain = MediaType.parseMediaType(PrometheusTextFormatWriter.CONTENT_TYPE); - client.get() - .uri(uri) - .accept(openMetrics, textPlain) - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(openMetrics); - } - - @WebEndpointTest - void scrapeWithIncludedNames(WebTestClient client) { - scrapeWithIncludedNames(client, "/actuator/prometheus?includedNames=counter1,counter2"); - scrapeWithIncludedNames(client, "/actuator/prometheussc?includedNames=counter1_total,counter2_total"); - } - - private void scrapeWithIncludedNames(WebTestClient client, String uri) { - client.get() - .uri(uri) - .exchange() - .expectStatus() - .isOk() - .expectHeader() - .contentType(MediaType.parseMediaType(PrometheusTextFormatWriter.CONTENT_TYPE)) - .expectBody(String.class) - .value((body) -> assertThat(body).contains("counter1_total") - .contains("counter2_total") - .doesNotContain("counter3_total")); - } - - @SuppressWarnings({ "deprecation", "removal" }) - @Configuration(proxyBeanMethods = false) - static class TestConfiguration { - - @Bean - PrometheusScrapeEndpoint prometheusScrapeEndpoint(PrometheusRegistry prometheusRegistry) { - return new PrometheusScrapeEndpoint(prometheusRegistry, new Properties()); - } - - @Bean - CustomPrometheusScrapeEndpoint customPrometheusScrapeEndpoint(CollectorRegistry collectorRegistry) { - return new CustomPrometheusScrapeEndpoint(collectorRegistry); - } - - @Bean - PrometheusRegistry prometheusRegistry() { - return new PrometheusRegistry(); - } - - @Bean - CollectorRegistry collectorRegistry() { - return new CollectorRegistry(true); - } - - @Bean - PrometheusMeterRegistry registry(PrometheusRegistry prometheusRegistry) { - return new PrometheusMeterRegistry((k) -> null, prometheusRegistry, Clock.SYSTEM); - } - - @Bean - io.micrometer.prometheus.PrometheusMeterRegistry oldRegistry(CollectorRegistry collectorRegistry) { - return new io.micrometer.prometheus.PrometheusMeterRegistry((k) -> null, collectorRegistry, Clock.SYSTEM); - } - - @Bean - CompositeMeterRegistry compositeMeterRegistry(PrometheusMeterRegistry prometheusMeterRegistry, - io.micrometer.prometheus.PrometheusMeterRegistry prometheusSCMeterRegistry) { - CompositeMeterRegistry composite = new CompositeMeterRegistry(); - composite.add(prometheusMeterRegistry).add(prometheusSCMeterRegistry); - Counter.builder("counter1").register(composite); - Counter.builder("counter2").register(composite); - Counter.builder("counter3").register(composite); - return composite; - } - - @WebEndpoint(id = "prometheussc") - static class CustomPrometheusScrapeEndpoint extends PrometheusSimpleclientScrapeEndpoint { - - CustomPrometheusScrapeEndpoint(CollectorRegistry collectorRegistry) { - super(collectorRegistry); - } - - } - - } - -} diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/quartz/QuartzEndpointTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/quartz/QuartzEndpointTests.java index 9fa95ed84ff0..f99f45cd48e5 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/quartz/QuartzEndpointTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/quartz/QuartzEndpointTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -66,6 +66,7 @@ import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzJobDetailsDescriptor; import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzJobGroupSummaryDescriptor; import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzJobSummaryDescriptor; +import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzJobTriggerDescriptor; import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzTriggerGroupSummaryDescriptor; import org.springframework.scheduling.quartz.DelegatingJob; import org.springframework.util.LinkedMultiValueMap; @@ -73,9 +74,12 @@ import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.entry; +import static org.assertj.core.api.Assertions.within; +import static org.mockito.ArgumentMatchers.any; import static org.mockito.BDDMockito.given; import static org.mockito.BDDMockito.then; import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.never; /** * Tests for {@link QuartzEndpoint}. @@ -755,6 +759,31 @@ void quartzJobWithDataMapAndShowUnsanitizedFalse() throws SchedulerException { entry("url", "******")); } + @Test + void quartzJobShouldBeTriggered() throws SchedulerException { + JobDetail job = JobBuilder.newJob(Job.class) + .withIdentity("hello", "samples") + .withDescription("A sample job") + .storeDurably() + .requestRecovery(false) + .build(); + mockJobs(job); + QuartzJobTriggerDescriptor quartzJobTriggerDescriptor = this.endpoint.triggerQuartzJob("samples", "hello"); + assertThat(quartzJobTriggerDescriptor).isNotNull(); + assertThat(quartzJobTriggerDescriptor.getName()).isEqualTo("hello"); + assertThat(quartzJobTriggerDescriptor.getGroup()).isEqualTo("samples"); + assertThat(quartzJobTriggerDescriptor.getClassName()).isEqualTo("org.quartz.Job"); + assertThat(quartzJobTriggerDescriptor.getTriggerTime()).isCloseTo(Instant.now(), within(5, ChronoUnit.SECONDS)); + then(this.scheduler).should().triggerJob(new JobKey("hello", "samples")); + } + + @Test + void quartzJobShouldNotBeTriggeredWhenJobDoesNotExist() throws SchedulerException { + QuartzJobTriggerDescriptor quartzJobTriggerDescriptor = this.endpoint.triggerQuartzJob("samples", "hello"); + assertThat(quartzJobTriggerDescriptor).isNull(); + then(this.scheduler).should(never()).triggerJob(any()); + } + private void mockJobs(JobDetail... jobs) throws SchedulerException { MultiValueMap jobKeys = new LinkedMultiValueMap<>(); for (JobDetail jobDetail : jobs) { diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/quartz/QuartzEndpointWebIntegrationTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/quartz/QuartzEndpointWebIntegrationTests.java index 907224e33cc3..e206046ec5b3 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/quartz/QuartzEndpointWebIntegrationTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/quartz/QuartzEndpointWebIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,6 +20,7 @@ import java.util.Collections; import java.util.LinkedHashSet; import java.util.List; +import java.util.Map; import java.util.Map.Entry; import net.minidev.json.JSONArray; @@ -46,6 +47,7 @@ import org.springframework.boot.actuate.endpoint.web.test.WebEndpointTest; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.http.MediaType; import org.springframework.scheduling.quartz.DelegatingJob; import org.springframework.test.web.reactive.server.WebTestClient; import org.springframework.util.LinkedMultiValueMap; @@ -249,6 +251,48 @@ void quartzTriggerDetailWithUnknownKey(WebTestClient client) { client.get().uri("/actuator/quartz/triggers/tests/does-not-exist").exchange().expectStatus().isNotFound(); } + @WebEndpointTest + void quartzTriggerJob(WebTestClient client) { + client.post() + .uri("/actuator/quartz/jobs/samples/jobOne") + .contentType(MediaType.APPLICATION_JSON) + .bodyValue(Map.of("state", "running")) + .exchange() + .expectStatus() + .isOk() + .expectBody() + .jsonPath("group") + .isEqualTo("samples") + .jsonPath("name") + .isEqualTo("jobOne") + .jsonPath("className") + .isEqualTo("org.quartz.Job") + .jsonPath("triggerTime") + .isNotEmpty(); + } + + @WebEndpointTest + void quartzTriggerJobWithUnknownJobKey(WebTestClient client) { + client.post() + .uri("/actuator/quartz/jobs/samples/does-not-exist") + .contentType(MediaType.APPLICATION_JSON) + .bodyValue(Map.of("state", "running")) + .exchange() + .expectStatus() + .isNotFound(); + } + + @WebEndpointTest + void quartzTriggerJobWithUnknownState(WebTestClient client) { + client.post() + .uri("/actuator/quartz/jobs/samples/jobOne") + .contentType(MediaType.APPLICATION_JSON) + .bodyValue(Map.of("state", "unknown")) + .exchange() + .expectStatus() + .isBadRequest(); + } + @Configuration(proxyBeanMethods = false) static class TestConfiguration { diff --git a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/web/mappings/MappingsEndpointTests.java b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/web/mappings/MappingsEndpointTests.java index 929b7a9e1ff7..5516fa69f5a4 100644 --- a/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/web/mappings/MappingsEndpointTests.java +++ b/spring-boot-project/spring-boot-actuator/src/test/java/org/springframework/boot/actuate/web/mappings/MappingsEndpointTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -57,6 +57,8 @@ import org.springframework.web.servlet.config.annotation.EnableWebMvc; import org.springframework.web.servlet.config.annotation.PathMatchConfigurer; import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; +import org.springframework.web.servlet.function.RequestPredicates; +import org.springframework.web.servlet.function.RouterFunctions; import org.springframework.web.util.pattern.PathPatternParser; import static org.assertj.core.api.Assertions.assertThat; @@ -71,6 +73,7 @@ * * @author Andy Wilkinson * @author Stephane Nicoll + * @author Xiong Tang */ class MappingsEndpointTests { @@ -88,7 +91,7 @@ void servletWebMappings() { "dispatcherServlets"); assertThat(dispatcherServlets).containsOnlyKeys("dispatcherServlet"); List handlerMappings = dispatcherServlets.get("dispatcherServlet"); - assertThat(handlerMappings).hasSize(1); + assertThat(handlerMappings).hasSize(4); List servlets = mappings(contextMappings, "servlets"); assertThat(servlets).hasSize(1); List filters = mappings(contextMappings, "servletFilters"); @@ -111,7 +114,7 @@ void servletWebMappingsWithPathPatternParser() { "dispatcherServlets"); assertThat(dispatcherServlets).containsOnlyKeys("dispatcherServlet"); List handlerMappings = dispatcherServlets.get("dispatcherServlet"); - assertThat(handlerMappings).hasSize(1); + assertThat(handlerMappings).hasSize(4); List servlets = mappings(contextMappings, "servlets"); assertThat(servlets).hasSize(1); List filters = mappings(contextMappings, "servletFilters"); @@ -131,9 +134,9 @@ void servletWebMappingsWithAdditionalDispatcherServlets() { "dispatcherServlets"); assertThat(dispatcherServlets).containsOnlyKeys("dispatcherServlet", "customDispatcherServletRegistration", "anotherDispatcherServletRegistration"); - assertThat(dispatcherServlets.get("dispatcherServlet")).hasSize(1); - assertThat(dispatcherServlets.get("customDispatcherServletRegistration")).hasSize(1); - assertThat(dispatcherServlets.get("anotherDispatcherServletRegistration")).hasSize(1); + assertThat(dispatcherServlets.get("dispatcherServlet")).hasSize(4); + assertThat(dispatcherServlets.get("customDispatcherServletRegistration")).hasSize(4); + assertThat(dispatcherServlets.get("anotherDispatcherServletRegistration")).hasSize(4); }); } @@ -248,11 +251,28 @@ DispatcherServlet dispatcherServlet(WebApplicationContext context) throws Servle return dispatcherServlet; } + @Bean + org.springframework.web.servlet.function.RouterFunction routerFunction() { + return RouterFunctions + .route(RequestPredicates.GET("/one"), + (request) -> org.springframework.web.servlet.function.ServerResponse.ok().build()) + .andRoute(RequestPredicates.POST("/two"), + (request) -> org.springframework.web.servlet.function.ServerResponse.ok().build()); + } + @RequestMapping("/three") void three() { } + @Bean + org.springframework.web.servlet.function.RouterFunction routerFunctionWithAttributes() { + return RouterFunctions + .route(RequestPredicates.GET("/four"), + (request) -> org.springframework.web.servlet.function.ServerResponse.ok().build()) + .withAttribute("test", "test"); + } + } @Configuration diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/AutoConfigurationImportSelector.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/AutoConfigurationImportSelector.java index e3da084cf8f9..0eba73f1dd95 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/AutoConfigurationImportSelector.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/AutoConfigurationImportSelector.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -57,6 +57,7 @@ import org.springframework.core.type.classreading.MetadataReaderFactory; import org.springframework.util.Assert; import org.springframework.util.ClassUtils; +import org.springframework.util.CollectionUtils; import org.springframework.util.StringUtils; /** @@ -170,7 +171,7 @@ protected boolean isEnabled(AnnotationMetadata metadata) { protected AnnotationAttributes getAttributes(AnnotationMetadata metadata) { String name = getAnnotationClass().getName(); AnnotationAttributes attributes = AnnotationAttributes.fromMap(metadata.getAnnotationAttributes(name, true)); - Assert.notNull(attributes, () -> "No auto-configuration attributes found. Is " + metadata.getClassName() + Assert.state(attributes != null, () -> "No auto-configuration attributes found. Is " + metadata.getClassName() + " annotated with " + ClassUtils.getShortName(name) + "?"); return attributes; } @@ -195,7 +196,7 @@ protected List getCandidateConfigurations(AnnotationMetadata metadata, A ImportCandidates importCandidates = ImportCandidates.load(this.autoConfigurationAnnotation, getBeanClassLoader()); List configurations = importCandidates.getCandidates(); - Assert.notEmpty(configurations, + Assert.state(!CollectionUtils.isEmpty(configurations), "No auto configuration classes found in " + "META-INF/spring/" + this.autoConfigurationAnnotation.getName() + ".imports. If you " + "are using a custom packaging, make sure that file is correct."); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/AutoConfigurationSorter.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/AutoConfigurationSorter.java index 043da47b27c3..4dee33cacd2a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/AutoConfigurationSorter.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/AutoConfigurationSorter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,7 +53,7 @@ class AutoConfigurationSorter { AutoConfigurationSorter(MetadataReaderFactory metadataReaderFactory, AutoConfigurationMetadata autoConfigurationMetadata, UnaryOperator replacementMapper) { - Assert.notNull(metadataReaderFactory, "MetadataReaderFactory must not be null"); + Assert.notNull(metadataReaderFactory, "'metadataReaderFactory' must not be null"); this.metadataReaderFactory = metadataReaderFactory; this.autoConfigurationMetadata = autoConfigurationMetadata; this.replacementMapper = replacementMapper; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/admin/SpringApplicationAdminJmxAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/admin/SpringApplicationAdminJmxAutoConfiguration.java index 6876fa4f402f..b6053bc088b3 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/admin/SpringApplicationAdminJmxAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/admin/SpringApplicationAdminJmxAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,8 +22,8 @@ import org.springframework.boot.admin.SpringApplicationAdminMXBean; import org.springframework.boot.admin.SpringApplicationAdminMXBeanRegistrar; import org.springframework.boot.autoconfigure.AutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.jmx.JmxAutoConfiguration; import org.springframework.context.annotation.Bean; import org.springframework.core.env.Environment; @@ -39,8 +39,7 @@ * @see SpringApplicationAdminMXBean */ @AutoConfiguration(after = JmxAutoConfiguration.class) -@ConditionalOnProperty(prefix = "spring.application.admin", value = "enabled", havingValue = "true", - matchIfMissing = false) +@ConditionalOnBooleanProperty("spring.application.admin.enabled") public class SpringApplicationAdminJmxAutoConfiguration { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/AbstractConnectionFactoryConfigurer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/AbstractConnectionFactoryConfigurer.java index d5f1a12c6666..dc346dddc3b2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/AbstractConnectionFactoryConfigurer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/AbstractConnectionFactoryConfigurer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,7 +48,7 @@ public abstract class AbstractConnectionFactoryConfigurer getAddresses() { return addresses; } + @Override + public SslBundle getSslBundle() { + Ssl ssl = this.properties.getSsl(); + if (!ssl.determineEnabled()) { + return null; + } + if (StringUtils.hasLength(ssl.getBundle())) { + Assert.notNull(this.sslBundles, "SSL bundle name has been set but no SSL bundles found in context"); + return this.sslBundles.getBundle(ssl.getBundle()); + } + return null; + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitAnnotationDrivenConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitAnnotationDrivenConfiguration.java index 65948817aae6..0a776b4df923 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitAnnotationDrivenConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitAnnotationDrivenConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -82,8 +82,7 @@ SimpleRabbitListenerContainerFactoryConfigurer simpleRabbitListenerContainerFact @Bean(name = "rabbitListenerContainerFactory") @ConditionalOnMissingBean(name = "rabbitListenerContainerFactory") - @ConditionalOnProperty(prefix = "spring.rabbitmq.listener", name = "type", havingValue = "simple", - matchIfMissing = true) + @ConditionalOnProperty(name = "spring.rabbitmq.listener.type", havingValue = "simple", matchIfMissing = true) SimpleRabbitListenerContainerFactory simpleRabbitListenerContainerFactory( SimpleRabbitListenerContainerFactoryConfigurer configurer, ConnectionFactory connectionFactory, ObjectProvider> simpleContainerCustomizer) { @@ -111,7 +110,7 @@ DirectRabbitListenerContainerFactoryConfigurer directRabbitListenerContainerFact @Bean(name = "rabbitListenerContainerFactory") @ConditionalOnMissingBean(name = "rabbitListenerContainerFactory") - @ConditionalOnProperty(prefix = "spring.rabbitmq.listener", name = "type", havingValue = "direct") + @ConditionalOnProperty(name = "spring.rabbitmq.listener.type", havingValue = "direct") DirectRabbitListenerContainerFactory directRabbitListenerContainerFactory( DirectRabbitListenerContainerFactoryConfigurer configurer, ConnectionFactory connectionFactory, ObjectProvider> directContainerCustomizer) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitAutoConfiguration.java index 49393474f65c..590ece485359 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,9 +33,9 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnSingleCandidate; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.boot.ssl.SslBundles; @@ -90,18 +90,17 @@ protected RabbitConnectionFactoryCreator(RabbitProperties properties) { @Bean @ConditionalOnMissingBean - RabbitConnectionDetails rabbitConnectionDetails() { - return new PropertiesRabbitConnectionDetails(this.properties); + RabbitConnectionDetails rabbitConnectionDetails(ObjectProvider sslBundles) { + return new PropertiesRabbitConnectionDetails(this.properties, sslBundles.getIfAvailable()); } @Bean @ConditionalOnMissingBean RabbitConnectionFactoryBeanConfigurer rabbitConnectionFactoryBeanConfigurer(ResourceLoader resourceLoader, RabbitConnectionDetails connectionDetails, ObjectProvider credentialsProvider, - ObjectProvider credentialsRefreshService, - ObjectProvider sslBundles) { + ObjectProvider credentialsRefreshService) { RabbitConnectionFactoryBeanConfigurer configurer = new RabbitConnectionFactoryBeanConfigurer(resourceLoader, - this.properties, connectionDetails, sslBundles.getIfAvailable()); + this.properties, connectionDetails); configurer.setCredentialsProvider(credentialsProvider.getIfUnique()); configurer.setCredentialsRefreshService(credentialsRefreshService.getIfUnique()); return configurer; @@ -164,7 +163,7 @@ public RabbitTemplate rabbitTemplate(RabbitTemplateConfigurer configurer, Connec @Bean @ConditionalOnSingleCandidate(ConnectionFactory.class) - @ConditionalOnProperty(prefix = "spring.rabbitmq", name = "dynamic", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.rabbitmq.dynamic", matchIfMissing = true) @ConditionalOnMissingBean public AmqpAdmin amqpAdmin(ConnectionFactory connectionFactory) { return new RabbitAdmin(connectionFactory); @@ -176,7 +175,7 @@ public AmqpAdmin amqpAdmin(ConnectionFactory connectionFactory) { @ConditionalOnClass(RabbitMessagingTemplate.class) @ConditionalOnMissingBean(RabbitMessagingTemplate.class) @Import(RabbitTemplateConfiguration.class) - protected static class MessagingTemplateConfiguration { + protected static class RabbitMessagingTemplateConfiguration { @Bean @ConditionalOnSingleCandidate(RabbitTemplate.class) diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitConnectionDetails.java index 4cdc51c2155e..ae4d595458ff 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,7 @@ import java.util.List; import org.springframework.boot.autoconfigure.service.connection.ConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.util.Assert; /** @@ -73,6 +74,15 @@ default Address getFirstAddress() { return addresses.get(0); } + /** + * SSL bundle to use. + * @return the SSL bundle to use + * @since 3.5.0 + */ + default SslBundle getSslBundle() { + return null; + } + /** * A RabbitMQ address. * diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitConnectionFactoryBeanConfigurer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitConnectionFactoryBeanConfigurer.java index 2f59e2d8faed..cc8a44ea0526 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitConnectionFactoryBeanConfigurer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitConnectionFactoryBeanConfigurer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,8 +48,6 @@ public class RabbitConnectionFactoryBeanConfigurer { private final RabbitConnectionDetails connectionDetails; - private final SslBundles sslBundles; - private CredentialsProvider credentialsProvider; private CredentialsRefreshService credentialsRefreshService; @@ -61,7 +59,7 @@ public class RabbitConnectionFactoryBeanConfigurer { * @param properties the properties */ public RabbitConnectionFactoryBeanConfigurer(ResourceLoader resourceLoader, RabbitProperties properties) { - this(resourceLoader, properties, new PropertiesRabbitConnectionDetails(properties)); + this(resourceLoader, properties, new PropertiesRabbitConnectionDetails(properties, null)); } /** @@ -90,13 +88,12 @@ public RabbitConnectionFactoryBeanConfigurer(ResourceLoader resourceLoader, Rabb */ public RabbitConnectionFactoryBeanConfigurer(ResourceLoader resourceLoader, RabbitProperties properties, RabbitConnectionDetails connectionDetails, SslBundles sslBundles) { - Assert.notNull(resourceLoader, "ResourceLoader must not be null"); - Assert.notNull(properties, "Properties must not be null"); - Assert.notNull(connectionDetails, "ConnectionDetails must not be null"); + Assert.notNull(resourceLoader, "'resourceLoader' must not be null"); + Assert.notNull(properties, "'properties' must not be null"); + Assert.notNull(connectionDetails, "'connectionDetails' must not be null"); this.resourceLoader = resourceLoader; this.rabbitProperties = properties; this.connectionDetails = connectionDetails; - this.sslBundles = sslBundles; } public void setCredentialsProvider(CredentialsProvider credentialsProvider) { @@ -115,7 +112,7 @@ public void setCredentialsRefreshService(CredentialsRefreshService credentialsRe * @param factory the {@link RabbitConnectionFactoryBean} instance to configure */ public void configure(RabbitConnectionFactoryBean factory) { - Assert.notNull(factory, "RabbitConnectionFactoryBean must not be null"); + Assert.notNull(factory, "'factory' must not be null"); factory.setResourceLoader(this.resourceLoader); Address address = this.connectionDetails.getFirstAddress(); PropertyMapper map = PropertyMapper.get(); @@ -129,16 +126,14 @@ public void configure(RabbitConnectionFactoryBean factory) { .asInt(Duration::getSeconds) .to(factory::setRequestedHeartbeat); map.from(this.rabbitProperties::getRequestedChannelMax).to(factory::setRequestedChannelMax); - RabbitProperties.Ssl ssl = this.rabbitProperties.getSsl(); - if (ssl.determineEnabled()) { - factory.setUseSSL(true); - if (ssl.getBundle() != null) { - SslBundle bundle = this.sslBundles.getBundle(ssl.getBundle()); - if (factory instanceof SslBundleRabbitConnectionFactoryBean sslFactory) { - sslFactory.setSslBundle(bundle); - } - } - else { + SslBundle sslBundle = this.connectionDetails.getSslBundle(); + if (sslBundle != null) { + applySslBundle(factory, sslBundle); + } + else { + RabbitProperties.Ssl ssl = this.rabbitProperties.getSsl(); + if (ssl.determineEnabled()) { + factory.setUseSSL(true); map.from(ssl::getAlgorithm).whenNonNull().to(factory::setSslAlgorithm); map.from(ssl::getKeyStoreType).to(factory::setKeyStoreType); map.from(ssl::getKeyStore).to(factory::setKeyStore); @@ -148,10 +143,10 @@ public void configure(RabbitConnectionFactoryBean factory) { map.from(ssl::getTrustStore).to(factory::setTrustStore); map.from(ssl::getTrustStorePassword).to(factory::setTrustStorePassphrase); map.from(ssl::getTrustStoreAlgorithm).whenNonNull().to(factory::setTrustStoreAlgorithm); + map.from(ssl::isValidateServerCertificate) + .to((validate) -> factory.setSkipServerCertificateValidation(!validate)); + map.from(ssl::isVerifyHostname).to(factory::setEnableHostnameVerification); } - map.from(ssl::isValidateServerCertificate) - .to((validate) -> factory.setSkipServerCertificateValidation(!validate)); - map.from(ssl::getVerifyHostname).to(factory::setEnableHostnameVerification); } map.from(this.rabbitProperties::getConnectionTimeout) .whenNonNull() @@ -169,4 +164,11 @@ public void configure(RabbitConnectionFactoryBean factory) { .to(factory::setMaxInboundMessageBodySize); } + private static void applySslBundle(RabbitConnectionFactoryBean factory, SslBundle bundle) { + factory.setUseSSL(true); + if (factory instanceof SslBundleRabbitConnectionFactoryBean sslFactory) { + sslFactory.setSslBundle(bundle); + } + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitProperties.java index 5d63ff2c8582..0b0b89a6a513 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitProperties.java @@ -50,7 +50,7 @@ * @author Yanming Zhou * @since 1.0.0 */ -@ConfigurationProperties(prefix = "spring.rabbitmq") +@ConfigurationProperties("spring.rabbitmq") public class RabbitProperties { private static final int DEFAULT_PORT = 5672; @@ -574,7 +574,7 @@ public void setValidateServerCertificate(boolean validateServerCertificate) { this.validateServerCertificate = validateServerCertificate; } - public boolean getVerifyHostname() { + public boolean isVerifyHostname() { return this.verifyHostname; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitStreamConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitStreamConfiguration.java index 6e3047d1dd53..2aad7eaebe78 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitStreamConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitStreamConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -52,7 +52,7 @@ class RabbitStreamConfiguration { @Bean(name = "rabbitListenerContainerFactory") @ConditionalOnMissingBean(name = "rabbitListenerContainerFactory") - @ConditionalOnProperty(prefix = "spring.rabbitmq.listener", name = "type", havingValue = "stream") + @ConditionalOnProperty(name = "spring.rabbitmq.listener.type", havingValue = "stream") StreamRabbitListenerContainerFactory streamRabbitListenerContainerFactory(Environment rabbitStreamEnvironment, RabbitProperties properties, ObjectProvider consumerCustomizer, ObjectProvider> containerCustomizer) { @@ -90,7 +90,7 @@ RabbitStreamTemplateConfigurer rabbitStreamTemplateConfigurer(RabbitProperties p @Bean @ConditionalOnMissingBean(RabbitStreamOperations.class) - @ConditionalOnProperty(prefix = "spring.rabbitmq.stream", name = "name") + @ConditionalOnProperty(name = "spring.rabbitmq.stream.name") RabbitStreamTemplate rabbitStreamTemplate(Environment rabbitStreamEnvironment, RabbitProperties properties, RabbitStreamTemplateConfigurer configurer) { RabbitStreamTemplate template = new RabbitStreamTemplate(rabbitStreamEnvironment, diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitTemplateConfigurer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitTemplateConfigurer.java index d002323c7416..668342908e31 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitTemplateConfigurer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/amqp/RabbitTemplateConfigurer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,7 @@ public class RabbitTemplateConfigurer { * @since 2.6.0 */ public RabbitTemplateConfigurer(RabbitProperties rabbitProperties) { - Assert.notNull(rabbitProperties, "RabbitProperties must not be null"); + Assert.notNull(rabbitProperties, "'rabbitProperties' must not be null"); this.rabbitProperties = rabbitProperties; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/aop/AopAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/aop/AopAutoConfiguration.java index 4767aac8b4a9..b0cd20c5590c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/aop/AopAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/aop/AopAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,9 +22,9 @@ import org.springframework.beans.factory.config.BeanFactoryPostProcessor; import org.springframework.beans.factory.support.BeanDefinitionRegistry; import org.springframework.boot.autoconfigure.AutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.EnableAspectJAutoProxy; @@ -44,7 +44,7 @@ * @see EnableAspectJAutoProxy */ @AutoConfiguration -@ConditionalOnProperty(prefix = "spring.aop", name = "auto", havingValue = "true", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.aop.auto", matchIfMissing = true) public class AopAutoConfiguration { @Configuration(proxyBeanMethods = false) @@ -53,15 +53,14 @@ static class AspectJAutoProxyingConfiguration { @Configuration(proxyBeanMethods = false) @EnableAspectJAutoProxy(proxyTargetClass = false) - @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class", havingValue = "false") + @ConditionalOnBooleanProperty(name = "spring.aop.proxy-target-class", havingValue = false) static class JdkDynamicAutoProxyConfiguration { } @Configuration(proxyBeanMethods = false) @EnableAspectJAutoProxy(proxyTargetClass = true) - @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class", havingValue = "true", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.aop.proxy-target-class", matchIfMissing = true) static class CglibAutoProxyConfiguration { } @@ -70,8 +69,7 @@ static class CglibAutoProxyConfiguration { @Configuration(proxyBeanMethods = false) @ConditionalOnMissingClass("org.aspectj.weaver.Advice") - @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class", havingValue = "true", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.aop.proxy-target-class", matchIfMissing = true) static class ClassProxyingConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfiguration.java index e09ab3e8de0e..c6891641091d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,6 +22,7 @@ import org.springframework.batch.core.configuration.annotation.EnableBatchProcessing; import org.springframework.batch.core.configuration.support.DefaultBatchConfiguration; +import org.springframework.batch.core.converter.JobParametersConverter; import org.springframework.batch.core.explore.JobExplorer; import org.springframework.batch.core.launch.JobLauncher; import org.springframework.batch.core.repository.ExecutionContextSerializer; @@ -31,9 +32,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration; import org.springframework.boot.autoconfigure.sql.init.OnDatabaseInitializationCondition; import org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration; @@ -66,6 +67,7 @@ * @author Mahmoud Ben Hassine * @author Lars Uffmann * @author Lasse Wulff + * @author Yanming Zhou * @since 1.0.0 */ @AutoConfiguration(after = { HibernateJpaAutoConfiguration.class, TransactionAutoConfiguration.class }) @@ -78,7 +80,7 @@ public class BatchAutoConfiguration { @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(prefix = "spring.batch.job", name = "enabled", havingValue = "true", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.batch.job.enabled", matchIfMissing = true) public JobLauncherApplicationRunner jobLauncherApplicationRunner(JobLauncher jobLauncher, JobExplorer jobExplorer, JobRepository jobRepository, BatchProperties properties) { JobLauncherApplicationRunner runner = new JobLauncherApplicationRunner(jobLauncher, jobExplorer, jobRepository); @@ -110,18 +112,22 @@ static class SpringBootBatchConfiguration extends DefaultBatchConfiguration { private final ExecutionContextSerializer executionContextSerializer; + private final JobParametersConverter jobParametersConverter; + SpringBootBatchConfiguration(DataSource dataSource, @BatchDataSource ObjectProvider batchDataSource, PlatformTransactionManager transactionManager, @BatchTransactionManager ObjectProvider batchTransactionManager, @BatchTaskExecutor ObjectProvider batchTaskExecutor, BatchProperties properties, ObjectProvider batchConversionServiceCustomizers, - ObjectProvider executionContextSerializer) { + ObjectProvider executionContextSerializer, + ObjectProvider jobParametersConverter) { this.dataSource = batchDataSource.getIfAvailable(() -> dataSource); this.transactionManager = batchTransactionManager.getIfAvailable(() -> transactionManager); this.taskExector = batchTaskExecutor.getIfAvailable(); this.properties = properties; this.batchConversionServiceCustomizers = batchConversionServiceCustomizers.orderedStream().toList(); this.executionContextSerializer = executionContextSerializer.getIfAvailable(); + this.jobParametersConverter = jobParametersConverter.getIfAvailable(); } @Override @@ -140,6 +146,11 @@ protected String getTablePrefix() { return (tablePrefix != null) ? tablePrefix : super.getTablePrefix(); } + @Override + protected boolean getValidateTransactionState() { + return this.properties.getJdbc().isValidateTransactionState(); + } + @Override protected Isolation getIsolationLevelForCreate() { Isolation isolation = this.properties.getJdbc().getIsolationLevelForCreate(); @@ -161,6 +172,12 @@ protected ExecutionContextSerializer getExecutionContextSerializer() { : super.getExecutionContextSerializer(); } + @Override + protected JobParametersConverter getJobParametersConverter() { + return (this.jobParametersConverter != null) ? this.jobParametersConverter + : super.getJobParametersConverter(); + } + @Override protected TaskExecutor getTaskExecutor() { return (this.taskExector != null) ? this.taskExector : super.getTaskExecutor(); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/BatchProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/BatchProperties.java index c308b01614ef..e12e4717560d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/BatchProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/BatchProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,9 +27,10 @@ * @author Eddú Meléndez * @author Vedran Pavic * @author Mukul Kumar Chaundhyan + * @author Yanming Zhou * @since 1.2.0 */ -@ConfigurationProperties(prefix = "spring.batch") +@ConfigurationProperties("spring.batch") public class BatchProperties { private final Job job = new Job(); @@ -67,6 +68,11 @@ public static class Jdbc { private static final String DEFAULT_SCHEMA_LOCATION = "classpath:org/springframework/" + "batch/core/schema-@@platform@@.sql"; + /** + * Whether to validate the transaction state. + */ + private boolean validateTransactionState = true; + /** * Transaction isolation level to use when creating job meta-data for new jobs. */ @@ -93,6 +99,14 @@ public static class Jdbc { */ private DatabaseInitializationMode initializeSchema = DatabaseInitializationMode.EMBEDDED; + public boolean isValidateTransactionState() { + return this.validateTransactionState; + } + + public void setValidateTransactionState(boolean validateTransactionState) { + this.validateTransactionState = validateTransactionState; + } + public Isolation getIsolationLevelForCreate() { return this.isolationLevelForCreate; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/JobLauncherApplicationRunner.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/JobLauncherApplicationRunner.java index a343346eb3e6..c96ef4e693d5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/JobLauncherApplicationRunner.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/batch/JobLauncherApplicationRunner.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -102,9 +102,9 @@ public class JobLauncherApplicationRunner * when running a job */ public JobLauncherApplicationRunner(JobLauncher jobLauncher, JobExplorer jobExplorer, JobRepository jobRepository) { - Assert.notNull(jobLauncher, "JobLauncher must not be null"); - Assert.notNull(jobExplorer, "JobExplorer must not be null"); - Assert.notNull(jobRepository, "JobRepository must not be null"); + Assert.notNull(jobLauncher, "'jobLauncher' must not be null"); + Assert.notNull(jobExplorer, "'jobExplorer' must not be null"); + Assert.notNull(jobRepository, "'jobRepository' must not be null"); this.jobLauncher = jobLauncher; this.jobExplorer = jobExplorer; this.jobRepository = jobRepository; @@ -112,10 +112,10 @@ public JobLauncherApplicationRunner(JobLauncher jobLauncher, JobExplorer jobExpl @Override public void afterPropertiesSet() { - Assert.isTrue(this.jobs.size() <= 1 || StringUtils.hasText(this.jobName), + Assert.state(this.jobs.size() <= 1 || StringUtils.hasText(this.jobName), "Job name must be specified in case of multiple jobs"); if (StringUtils.hasText(this.jobName)) { - Assert.isTrue(isLocalJob(this.jobName) || isRegisteredJob(this.jobName), + Assert.state(isLocalJob(this.jobName) || isRegisteredJob(this.jobName), () -> "No job found with name '" + this.jobName + "'"); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cache/CacheAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cache/CacheAutoConfiguration.java index d41c4f434545..49d331a58f61 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cache/CacheAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cache/CacheAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -102,7 +102,7 @@ static class CacheManagerValidator implements InitializingBean { @Override public void afterPropertiesSet() { - Assert.notNull(this.cacheManager.getIfAvailable(), + Assert.state(this.cacheManager.getIfAvailable() != null, () -> "No cache manager could be auto-configured, check your configuration (caching type is '" + this.cacheProperties.getType() + "')"); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cache/CacheProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cache/CacheProperties.java index 62ce77a7184c..aee56be51515 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cache/CacheProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cache/CacheProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ * @author Ryon Day * @since 1.3.0 */ -@ConfigurationProperties(prefix = "spring.cache") +@ConfigurationProperties("spring.cache") public class CacheProperties { /** @@ -102,7 +102,7 @@ public Redis getRedis() { public Resource resolveConfigLocation(Resource config) { if (config != null) { Assert.isTrue(config.exists(), - () -> "Cache configuration does not exist '" + config.getDescription() + "'"); + () -> "'config' resource [%s] must exist".formatted(config.getDescription())); return config; } return null; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraAutoConfiguration.java index 7d0f78be8cf4..a666296324f4 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,15 +17,12 @@ package org.springframework.boot.autoconfigure.cassandra; import java.io.IOException; -import java.security.NoSuchAlgorithmException; import java.time.Duration; import java.util.Collections; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; -import javax.net.ssl.SSLContext; - import com.datastax.oss.driver.api.core.CqlSession; import com.datastax.oss.driver.api.core.CqlSessionBuilder; import com.datastax.oss.driver.api.core.config.DefaultDriverOption; @@ -99,8 +96,8 @@ public class CassandraAutoConfiguration { @Bean @ConditionalOnMissingBean(CassandraConnectionDetails.class) - PropertiesCassandraConnectionDetails cassandraConnectionDetails() { - return new PropertiesCassandraConnectionDetails(this.properties); + PropertiesCassandraConnectionDetails cassandraConnectionDetails(ObjectProvider sslBundles) { + return new PropertiesCassandraConnectionDetails(this.properties, sslBundles.getIfAvailable()); } @Bean @@ -115,10 +112,10 @@ public CqlSession cassandraSession(CqlSessionBuilder cqlSessionBuilder) { @Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE) public CqlSessionBuilder cassandraSessionBuilder(DriverConfigLoader driverConfigLoader, CassandraConnectionDetails connectionDetails, - ObjectProvider builderCustomizers, ObjectProvider sslBundles) { + ObjectProvider builderCustomizers) { CqlSessionBuilder builder = CqlSession.builder().withConfigLoader(driverConfigLoader); configureAuthentication(builder, connectionDetails); - configureSsl(builder, sslBundles.getIfAvailable()); + configureSsl(builder, connectionDetails); builder.withKeyspace(this.properties.getKeyspaceName()); builderCustomizers.orderedStream().forEach((customizer) -> customizer.customize(builder)); return builder; @@ -131,30 +128,11 @@ private void configureAuthentication(CqlSessionBuilder builder, CassandraConnect } } - private void configureSsl(CqlSessionBuilder builder, SslBundles sslBundles) { - Ssl properties = this.properties.getSsl(); - if (properties == null || !properties.isEnabled()) { + private void configureSsl(CqlSessionBuilder builder, CassandraConnectionDetails connectionDetails) { + SslBundle sslBundle = connectionDetails.getSslBundle(); + if (sslBundle == null) { return; } - String bundleName = properties.getBundle(); - if (!StringUtils.hasLength(bundleName)) { - configureDefaultSslContext(builder); - } - else { - configureSsl(builder, sslBundles.getBundle(bundleName)); - } - } - - private void configureDefaultSslContext(CqlSessionBuilder builder) { - try { - builder.withSslContext(SSLContext.getDefault()); - } - catch (NoSuchAlgorithmException ex) { - throw new IllegalStateException("Could not setup SSL default context for Cassandra", ex); - } - } - - private void configureSsl(CqlSessionBuilder builder, SslBundle sslBundle) { SslOptions options = sslBundle.getOptions(); Assert.state(options.getEnabledProtocols() == null, "SSL protocol options cannot be specified with Cassandra"); builder @@ -320,8 +298,11 @@ static final class PropertiesCassandraConnectionDetails implements CassandraConn private final CassandraProperties properties; - private PropertiesCassandraConnectionDetails(CassandraProperties properties) { + private final SslBundles sslBundles; + + private PropertiesCassandraConnectionDetails(CassandraProperties properties, SslBundles sslBundles) { this.properties = properties; + this.sslBundles = sslBundles; } @Override @@ -346,6 +327,19 @@ public String getLocalDatacenter() { return this.properties.getLocalDatacenter(); } + @Override + public SslBundle getSslBundle() { + Ssl ssl = this.properties.getSsl(); + if (ssl == null || !ssl.isEnabled()) { + return null; + } + if (StringUtils.hasLength(ssl.getBundle())) { + Assert.notNull(this.sslBundles, "SSL bundle name has been set but no SSL bundles found in context"); + return this.sslBundles.getBundle(ssl.getBundle()); + } + return SslBundle.systemDefault(); + } + private Node asNode(String contactPoint) { int i = contactPoint.lastIndexOf(':'); if (i >= 0) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraConnectionDetails.java index b02972269a05..e98961f6ef3d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,7 @@ import java.util.List; import org.springframework.boot.autoconfigure.service.connection.ConnectionDetails; +import org.springframework.boot.ssl.SslBundle; /** * Details required to establish a connection to a Cassandra service. @@ -59,6 +60,15 @@ default String getPassword() { */ String getLocalDatacenter(); + /** + * SSL bundle to use. + * @return the SSL bundle to use + * @since 3.5.0 + */ + default SslBundle getSslBundle() { + return null; + } + /** * A Cassandra node. * diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraProperties.java index f2f8cae7e39c..2e6762fe3a23 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/cassandra/CassandraProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * @author Scott Frederick * @since 1.3.0 */ -@ConfigurationProperties(prefix = "spring.cassandra") +@ConfigurationProperties("spring.cassandra") public class CassandraProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/codec/CodecProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/codec/CodecProperties.java index 3686d39c369e..9f8403908e86 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/codec/CodecProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/codec/CodecProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,7 +16,9 @@ package org.springframework.boot.autoconfigure.codec; +import org.springframework.boot.autoconfigure.http.codec.HttpCodecsProperties; import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.context.properties.DeprecatedConfigurationProperty; import org.springframework.util.unit.DataSize; /** @@ -24,8 +26,10 @@ * * @author Brian Clozel * @since 2.2.1 + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of {@link HttpCodecsProperties} */ -@ConfigurationProperties(prefix = "spring.codec") +@ConfigurationProperties("spring.codec") +@Deprecated(since = "3.5.0", forRemoval = true) public class CodecProperties { /** @@ -41,6 +45,7 @@ public class CodecProperties { */ private DataSize maxInMemorySize; + @DeprecatedConfigurationProperty(since = "3.5.0", replacement = "spring.http.codec.log-request-details") public boolean isLogRequestDetails() { return this.logRequestDetails; } @@ -49,6 +54,7 @@ public void setLogRequestDetails(boolean logRequestDetails) { this.logRequestDetails = logRequestDetails; } + @DeprecatedConfigurationProperty(since = "3.5.0", replacement = "spring.http.codec.max-in-memory-size") public DataSize getMaxInMemorySize() { return this.maxInMemorySize; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/AbstractNestedCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/AbstractNestedCondition.java index 1a317b0b5fe7..82a8d5460a76 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/AbstractNestedCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/AbstractNestedCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ public abstract class AbstractNestedCondition extends SpringBootCondition implem private final ConfigurationPhase configurationPhase; AbstractNestedCondition(ConfigurationPhase configurationPhase) { - Assert.notNull(configurationPhase, "ConfigurationPhase must not be null"); + Assert.notNull(configurationPhase, "'configurationPhase' must not be null"); this.configurationPhase = configurationPhase; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionEvaluationReport.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionEvaluationReport.java index 3126167f106e..5b87e4553eda 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionEvaluationReport.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionEvaluationReport.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,6 +27,8 @@ import java.util.Set; import java.util.SortedMap; import java.util.TreeMap; +import java.util.stream.Stream; +import java.util.stream.StreamSupport; import org.springframework.beans.factory.BeanFactory; import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; @@ -76,9 +78,9 @@ private ConditionEvaluationReport() { * @param outcome the condition outcome */ public void recordConditionEvaluation(String source, Condition condition, ConditionOutcome outcome) { - Assert.notNull(source, "Source must not be null"); - Assert.notNull(condition, "Condition must not be null"); - Assert.notNull(outcome, "Outcome must not be null"); + Assert.notNull(source, "'source' must not be null"); + Assert.notNull(condition, "'condition' must not be null"); + Assert.notNull(outcome, "'outcome' must not be null"); this.unconditionalClasses.remove(source); this.outcomes.computeIfAbsent(source, (key) -> new ConditionAndOutcomes()).add(condition, outcome); this.addedAncestorOutcomes = false; @@ -89,7 +91,7 @@ public void recordConditionEvaluation(String source, Condition condition, Condit * @param exclusions the names of the excluded classes */ public void recordExclusions(Collection exclusions) { - Assert.notNull(exclusions, "exclusions must not be null"); + Assert.notNull(exclusions, "'exclusions' must not be null"); this.exclusions.addAll(exclusions); } @@ -99,7 +101,7 @@ public void recordExclusions(Collection exclusions) { * evaluated */ public void recordEvaluationCandidates(List evaluationCandidates) { - Assert.notNull(evaluationCandidates, "evaluationCandidates must not be null"); + Assert.notNull(evaluationCandidates, "'evaluationCandidates' must not be null"); this.unconditionalClasses.addAll(evaluationCandidates); } @@ -237,6 +239,15 @@ public boolean isFullMatch() { return true; } + /** + * Return a {@link Stream} of the {@link ConditionAndOutcome} items. + * @return a stream of the {@link ConditionAndOutcome} items. + * @since 3.5.0 + */ + public Stream stream() { + return StreamSupport.stream(spliterator(), false); + } + @Override public Iterator iterator() { return Collections.unmodifiableSet(this.outcomes).iterator(); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionMessage.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionMessage.java index 9b0a0a40a170..1f4e2c898a3f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionMessage.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionMessage.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -108,7 +108,7 @@ public ConditionMessage append(String message) { * @see #forCondition(Class, Object...) */ public Builder andCondition(Class condition, Object... details) { - Assert.notNull(condition, "Condition must not be null"); + Assert.notNull(condition, "'condition' must not be null"); return andCondition("@" + ClassUtils.getShortName(condition), details); } @@ -122,7 +122,7 @@ public Builder andCondition(Class condition, Object... det * @see #forCondition(String, Object...) */ public Builder andCondition(String condition, Object... details) { - Assert.notNull(condition, "Condition must not be null"); + Assert.notNull(condition, "'condition' must not be null"); String detail = StringUtils.arrayToDelimitedString(details, " "); if (StringUtils.hasLength(detail)) { return new Builder(condition + " " + detail); @@ -379,7 +379,7 @@ public ConditionMessage items(Collection items) { * @return a built {@link ConditionMessage} */ public ConditionMessage items(Style style, Collection items) { - Assert.notNull(style, "Style must not be null"); + Assert.notNull(style, "'style' must not be null"); StringBuilder message = new StringBuilder(this.reason); items = style.applyTo(items); if ((this.condition == null || items == null || items.size() <= 1) diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionOutcome.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionOutcome.java index ebaa1ae33b6b..4ff371300b1b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionOutcome.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionOutcome.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,7 +48,7 @@ public ConditionOutcome(boolean match, String message) { * @param message the condition message */ public ConditionOutcome(boolean match, ConditionMessage message) { - Assert.notNull(message, "ConditionMessage must not be null"); + Assert.notNull(message, "'message' must not be null"); this.match = match; this.message = message; } @@ -153,7 +153,10 @@ public String toString() { * @param outcome the outcome to inverse * @return the inverse of the condition outcome * @since 1.3.0 + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of + * {@link #ConditionOutcome(boolean, ConditionMessage)} */ + @Deprecated(since = "3.5.0", forRemoval = true) public static ConditionOutcome inverse(ConditionOutcome outcome) { return new ConditionOutcome(!outcome.isMatch(), outcome.getConditionMessage()); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBooleanProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBooleanProperties.java new file mode 100644 index 000000000000..7bc923fc21e1 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBooleanProperties.java @@ -0,0 +1,48 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.condition; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.context.annotation.Conditional; + +/** + * Container annotation that aggregates several + * {@link ConditionalOnBooleanProperty @ConditionalOnBooleanProperty} annotations. + * + * @author Phillip Webb + * @since 3.5.0 + * @see ConditionalOnBooleanProperty + */ +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.METHOD }) +@Documented +@Conditional(OnPropertyCondition.class) +public @interface ConditionalOnBooleanProperties { + + /** + * Return the contained + * {@link ConditionalOnBooleanProperty @ConditionalOnBooleanProperty} annotations. + * @return the contained annotations + */ + ConditionalOnBooleanProperty[] value(); + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBooleanProperty.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBooleanProperty.java new file mode 100644 index 000000000000..a53e95aee689 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBooleanProperty.java @@ -0,0 +1,93 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.condition; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Repeatable; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.context.annotation.Conditional; +import org.springframework.core.env.Environment; + +/** + * {@link Conditional @Conditional} that checks if the specified properties have a + * specific boolean value. By default the properties must be present in the + * {@link Environment} and equal to {@code true}. The {@link #havingValue()} and + * {@link #matchIfMissing()} attributes allow further customizations. + *

+ * If the property is not contained in the {@link Environment} at all, the + * {@link #matchIfMissing()} attribute is consulted. By default missing attributes do not + * match. + * + * @author Phillip Webb + * @since 3.5.0 + * @see ConditionalOnProperty + */ +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.METHOD }) +@Documented +@Conditional(OnPropertyCondition.class) +@Repeatable(ConditionalOnBooleanProperties.class) +public @interface ConditionalOnBooleanProperty { + + /** + * Alias for {@link #name()}. + * @return the names + */ + String[] value() default {}; + + /** + * A prefix that should be applied to each property. The prefix automatically ends + * with a dot if not specified. A valid prefix is defined by one or more words + * separated with dots (e.g. {@code "acme.system.feature"}). + * @return the prefix + */ + String prefix() default ""; + + /** + * The name of the properties to test. If a prefix has been defined, it is applied to + * compute the full key of each property. For instance if the prefix is + * {@code app.config} and one value is {@code my-value}, the full key would be + * {@code app.config.my-value} + *

+ * Use the dashed notation to specify each property, that is all lower case with a "-" + * to separate words (e.g. {@code my-long-property}). + *

+ * If multiple names are specified, all of the properties have to pass the test for + * the condition to match. + * @return the names + */ + String[] name() default {}; + + /** + * The expected value for the properties. If not specified, the property must be equal + * to {@code true}. + * @return the expected value + */ + boolean havingValue() default true; + + /** + * Specify if the condition should match if the property is not set. Defaults to + * {@code false}. + * @return if the condition should match if the property is missing + */ + boolean matchIfMissing() default false; + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnProperties.java new file mode 100644 index 000000000000..c93a7b4ea906 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnProperties.java @@ -0,0 +1,48 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.condition; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.context.annotation.Conditional; + +/** + * Container annotation that aggregates several + * {@link ConditionalOnProperty @ConditionalOnProperty} annotations. + * + * @author Phillip Webb + * @since 3.5.0 + * @see ConditionalOnProperty + */ +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.METHOD }) +@Documented +@Conditional(OnPropertyCondition.class) +public @interface ConditionalOnProperties { + + /** + * Return the contained {@link ConditionalOnProperty @ConditionalOnProperty} + * annotations. + * @return the contained annotations + */ + ConditionalOnProperty[] value(); + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnProperty.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnProperty.java index 8bd4647a0cde..3b72400a4b74 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnProperty.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/ConditionalOnProperty.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,6 +18,7 @@ import java.lang.annotation.Documented; import java.lang.annotation.ElementType; +import java.lang.annotation.Repeatable; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @@ -88,11 +89,13 @@ * @author Stephane Nicoll * @author Phillip Webb * @since 1.1.0 + * @see ConditionalOnBooleanProperty */ @Retention(RetentionPolicy.RUNTIME) @Target({ ElementType.TYPE, ElementType.METHOD }) @Documented @Conditional(OnPropertyCondition.class) +@Repeatable(ConditionalOnProperties.class) public @interface ConditionalOnProperty { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnBeanCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnBeanCondition.java index 25ef759f726f..607fe647843a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnBeanCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnBeanCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -76,6 +76,7 @@ * @author Jakub Kubrynski * @author Stephane Nicoll * @author Andy Wilkinson + * @author Uladzislau Seuruk * @see ConditionalOnBean * @see ConditionalOnMissingBean * @see ConditionalOnSingleCandidate @@ -205,13 +206,13 @@ protected final MatchResult getMatchingBeans(Spec spec) { ConfigurableListableBeanFactory beanFactory = getSearchBeanFactory(spec); ClassLoader classLoader = spec.getContext().getClassLoader(); boolean considerHierarchy = spec.getStrategy() != SearchStrategy.CURRENT; - Set> parameterizedContainers = spec.getParameterizedContainers(); + Set parameterizedContainers = spec.getParameterizedContainers(); MatchResult result = new MatchResult(); - Set beansIgnoredByType = getNamesOfBeansIgnoredByType(classLoader, beanFactory, considerHierarchy, + Set beansIgnoredByType = getNamesOfBeansIgnoredByType(beanFactory, considerHierarchy, spec.getIgnoredTypes(), parameterizedContainers); - for (String type : spec.getTypes()) { - Map typeMatchedDefinitions = getBeanDefinitionsForType(classLoader, - considerHierarchy, beanFactory, type, parameterizedContainers); + for (ResolvableType type : spec.getTypes()) { + Map typeMatchedDefinitions = getBeanDefinitionsForType(beanFactory, + considerHierarchy, type, parameterizedContainers); Set typeMatchedNames = matchedNamesFrom(typeMatchedDefinitions, (name, definition) -> !ScopedProxyUtils.isScopedTarget(name) && isCandidate(beanFactory, name, definition, beansIgnoredByType)); @@ -250,8 +251,8 @@ private ConfigurableListableBeanFactory getSearchBeanFactory(Spec spec) { ConfigurableListableBeanFactory beanFactory = spec.getContext().getBeanFactory(); if (spec.getStrategy() == SearchStrategy.ANCESTORS) { BeanFactory parent = beanFactory.getParentBeanFactory(); - Assert.isInstanceOf(ConfigurableListableBeanFactory.class, parent, - "Unable to use SearchStrategy.ANCESTORS"); + Assert.state(parent instanceof ConfigurableListableBeanFactory, + "Unable to use SearchStrategy.ANCESTORS without ConfigurableListableBeanFactory"); beanFactory = (ConfigurableListableBeanFactory) parent; } return beanFactory; @@ -296,42 +297,31 @@ private boolean isDefaultCandidate(BeanDefinition definition) { return true; } - private Set getNamesOfBeansIgnoredByType(ClassLoader classLoader, ListableBeanFactory beanFactory, - boolean considerHierarchy, Set ignoredTypes, Set> parameterizedContainers) { + private Set getNamesOfBeansIgnoredByType(ListableBeanFactory beanFactory, boolean considerHierarchy, + Set ignoredTypes, Set parameterizedContainers) { Set result = null; - for (String ignoredType : ignoredTypes) { - Collection ignoredNames = getBeanDefinitionsForType(classLoader, considerHierarchy, beanFactory, - ignoredType, parameterizedContainers) + for (ResolvableType ignoredType : ignoredTypes) { + Collection ignoredNames = getBeanDefinitionsForType(beanFactory, considerHierarchy, ignoredType, + parameterizedContainers) .keySet(); result = addAll(result, ignoredNames); } return (result != null) ? result : Collections.emptySet(); } - private Map getBeanDefinitionsForType(ClassLoader classLoader, boolean considerHierarchy, - ListableBeanFactory beanFactory, String type, Set> parameterizedContainers) throws LinkageError { - try { - return getBeanDefinitionsForType(beanFactory, considerHierarchy, resolve(type, classLoader), - parameterizedContainers); - } - catch (ClassNotFoundException | NoClassDefFoundError ex) { - return Collections.emptyMap(); - } - } - private Map getBeanDefinitionsForType(ListableBeanFactory beanFactory, - boolean considerHierarchy, Class type, Set> parameterizedContainers) { + boolean considerHierarchy, ResolvableType type, Set parameterizedContainers) { Map result = collectBeanDefinitionsForType(beanFactory, considerHierarchy, type, parameterizedContainers, null); return (result != null) ? result : Collections.emptyMap(); } private Map collectBeanDefinitionsForType(ListableBeanFactory beanFactory, - boolean considerHierarchy, Class type, Set> parameterizedContainers, + boolean considerHierarchy, ResolvableType type, Set parameterizedContainers, Map result) { result = putAll(result, beanFactory.getBeanNamesForType(type, true, false), beanFactory); - for (Class container : parameterizedContainers) { - ResolvableType generic = ResolvableType.forClassWithGenerics(container, type); + for (ResolvableType parameterizedContainer : parameterizedContainers) { + ResolvableType generic = ResolvableType.forClassWithGenerics(parameterizedContainer.resolve(), type); result = putAll(result, beanFactory.getBeanNamesForType(generic, true, false), beanFactory); } if (considerHierarchy && beanFactory instanceof HierarchicalBeanFactory hierarchicalBeanFactory) { @@ -550,13 +540,13 @@ private static class Spec { private final Set names; - private final Set types; + private final Set types; private final Set annotations; - private final Set ignoredTypes; + private final Set ignoredTypes; - private final Set> parameterizedContainers; + private final Set parameterizedContainers; private final SearchStrategy strategy; @@ -570,10 +560,10 @@ private static class Spec { this.annotationType = annotationType; this.names = extract(attributes, "name"); this.annotations = extract(attributes, "annotation"); - this.ignoredTypes = extract(attributes, "ignored", "ignoredType"); + this.ignoredTypes = resolveWhenPossible(extract(attributes, "ignored", "ignoredType")); this.parameterizedContainers = resolveWhenPossible(extract(attributes, "parameterizedContainer")); this.strategy = annotation.getValue("search", SearchStrategy.class).orElse(null); - Set types = extractTypes(attributes); + Set types = resolveWhenPossible(extractTypes(attributes)); BeanTypeDeductionException deductionException = null; if (types.isEmpty() && this.names.isEmpty() && this.annotations.isEmpty()) { try { @@ -614,17 +604,18 @@ private void merge(Set result, String... additional) { Collections.addAll(result, additional); } - private Set> resolveWhenPossible(Set classNames) { + private Set resolveWhenPossible(Set classNames) { if (classNames.isEmpty()) { return Collections.emptySet(); } - Set> resolved = new LinkedHashSet<>(classNames.size()); + Set resolved = new LinkedHashSet<>(classNames.size()); for (String className : classNames) { try { - resolved.add(resolve(className, this.context.getClassLoader())); + Class type = resolve(className, this.context.getClassLoader()); + resolved.add(ResolvableType.forRawClass(type)); } catch (ClassNotFoundException | NoClassDefFoundError ex) { - // Ignore + resolved.add(ResolvableType.NONE); } } return resolved; @@ -653,48 +644,44 @@ protected final String getAnnotationName() { return "@" + ClassUtils.getShortName(this.annotationType); } - private Set deducedBeanType(ConditionContext context, AnnotatedTypeMetadata metadata) { + private Set deducedBeanType(ConditionContext context, AnnotatedTypeMetadata metadata) { if (metadata instanceof MethodMetadata && metadata.isAnnotated(Bean.class.getName())) { return deducedBeanTypeForBeanMethod(context, (MethodMetadata) metadata); } return Collections.emptySet(); } - private Set deducedBeanTypeForBeanMethod(ConditionContext context, MethodMetadata metadata) { + private Set deducedBeanTypeForBeanMethod(ConditionContext context, MethodMetadata metadata) { try { - Class returnType = getReturnType(context, metadata); - return Collections.singleton(returnType.getName()); + return Set.of(getReturnType(context, metadata)); } catch (Throwable ex) { throw new BeanTypeDeductionException(metadata.getDeclaringClassName(), metadata.getMethodName(), ex); } } - private Class getReturnType(ConditionContext context, MethodMetadata metadata) + private ResolvableType getReturnType(ConditionContext context, MethodMetadata metadata) throws ClassNotFoundException, LinkageError { // Safe to load at this point since we are in the REGISTER_BEAN phase ClassLoader classLoader = context.getClassLoader(); - Class returnType = resolve(metadata.getReturnTypeName(), classLoader); - if (isParameterizedContainer(returnType)) { - returnType = getReturnTypeGeneric(metadata, classLoader); + ResolvableType returnType = getMethodReturnType(metadata, classLoader); + if (isParameterizedContainer(returnType.resolve())) { + returnType = returnType.getGeneric(); } return returnType; } private boolean isParameterizedContainer(Class type) { - for (Class parameterizedContainer : this.parameterizedContainers) { - if (parameterizedContainer.isAssignableFrom(type)) { - return true; - } - } - return false; + return (type != null) && this.parameterizedContainers.stream() + .map(ResolvableType::resolve) + .anyMatch((container) -> container != null && container.isAssignableFrom(type)); } - private Class getReturnTypeGeneric(MethodMetadata metadata, ClassLoader classLoader) + private ResolvableType getMethodReturnType(MethodMetadata metadata, ClassLoader classLoader) throws ClassNotFoundException, LinkageError { Class declaringClass = resolve(metadata.getDeclaringClassName(), classLoader); Method beanMethod = findBeanMethod(declaringClass, metadata.getMethodName()); - return ResolvableType.forMethodReturnType(beanMethod).resolveGeneric(); + return ResolvableType.forMethodReturnType(beanMethod); } private Method findBeanMethod(Class declaringClass, String methodName) { @@ -720,6 +707,10 @@ private SearchStrategy getStrategy() { return (this.strategy != null) ? this.strategy : SearchStrategy.ALL; } + Set getTypes() { + return this.types; + } + private ConditionContext getContext() { return this.context; } @@ -728,19 +719,15 @@ private Set getNames() { return this.names; } - protected Set getTypes() { - return this.types; - } - private Set getAnnotations() { return this.annotations; } - private Set getIgnoredTypes() { + private Set getIgnoredTypes() { return this.ignoredTypes; } - private Set> getParameterizedContainers() { + private Set getParameterizedContainers() { return this.parameterizedContainers; } @@ -847,13 +834,13 @@ private void recordUnmatchedAnnotation(String annotation) { this.unmatchedAnnotations.add(annotation); } - private void recordMatchedType(String type, Collection matchingNames) { - this.matchedTypes.put(type, matchingNames); + private void recordMatchedType(ResolvableType type, Collection matchingNames) { + this.matchedTypes.put(type.toString(), matchingNames); this.namesOfAllMatches.addAll(matchingNames); } - private void recordUnmatchedType(String type) { - this.unmatchedTypes.add(type); + private void recordUnmatchedType(ResolvableType type) { + this.unmatchedTypes.add(type.toString()); } boolean isAllMatched() { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnPropertyCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnPropertyCondition.java index 5af58096ed80..c502fa9d0b38 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnPropertyCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnPropertyCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,9 +16,11 @@ package org.springframework.boot.autoconfigure.condition; +import java.lang.annotation.Annotation; import java.util.ArrayList; +import java.util.Arrays; import java.util.List; -import java.util.Map; +import java.util.stream.Stream; import org.springframework.boot.autoconfigure.condition.ConditionMessage.Style; import org.springframework.context.annotation.Condition; @@ -27,10 +29,12 @@ import org.springframework.core.annotation.AnnotationAttributes; import org.springframework.core.annotation.MergedAnnotation; import org.springframework.core.annotation.MergedAnnotationPredicates; +import org.springframework.core.annotation.MergedAnnotations; import org.springframework.core.annotation.Order; import org.springframework.core.env.PropertyResolver; import org.springframework.core.type.AnnotatedTypeMetadata; import org.springframework.util.Assert; +import org.springframework.util.ClassUtils; import org.springframework.util.StringUtils; /** @@ -41,21 +45,19 @@ * @author Stephane Nicoll * @author Andy Wilkinson * @see ConditionalOnProperty + * @see ConditionalOnBooleanProperty */ @Order(Ordered.HIGHEST_PRECEDENCE + 40) class OnPropertyCondition extends SpringBootCondition { @Override public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeMetadata metadata) { - List allAnnotationAttributes = metadata.getAnnotations() - .stream(ConditionalOnProperty.class.getName()) - .filter(MergedAnnotationPredicates.unique(MergedAnnotation::getMetaTypes)) - .map(MergedAnnotation::asAnnotationAttributes) - .toList(); + MergedAnnotations mergedAnnotations = metadata.getAnnotations(); + List> annotations = stream(mergedAnnotations).toList(); List noMatch = new ArrayList<>(); List match = new ArrayList<>(); - for (AnnotationAttributes annotationAttributes : allAnnotationAttributes) { - ConditionOutcome outcome = determineOutcome(annotationAttributes, context.getEnvironment()); + for (MergedAnnotation annotation : annotations) { + ConditionOutcome outcome = determineOutcome(annotation, context.getEnvironment()); (outcome.isMatch() ? match : noMatch).add(outcome.getConditionMessage()); } if (!noMatch.isEmpty()) { @@ -64,53 +66,90 @@ public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeM return ConditionOutcome.match(ConditionMessage.of(match)); } - private ConditionOutcome determineOutcome(AnnotationAttributes annotationAttributes, PropertyResolver resolver) { - Spec spec = new Spec(annotationAttributes); + private Stream> stream(MergedAnnotations mergedAnnotations) { + return Stream.concat(stream(mergedAnnotations, ConditionalOnProperty.class, ConditionalOnProperties.class), + stream(mergedAnnotations, ConditionalOnBooleanProperty.class, ConditionalOnBooleanProperties.class)); + } + + private Stream> stream(MergedAnnotations mergedAnnotations, + Class type, Class containerType) { + return Stream.concat(stream(mergedAnnotations, type), streamRepeated(mergedAnnotations, type, containerType)); + } + + private Stream> streamRepeated(MergedAnnotations mergedAnnotations, + Class type, Class containerType) { + return stream(mergedAnnotations, containerType).flatMap((container) -> streamRepeated(container, type)); + } + + @SuppressWarnings("unchecked") + private Stream> streamRepeated(MergedAnnotation container, + Class type) { + MergedAnnotation[] repeated = container.getAnnotationArray(MergedAnnotation.VALUE, type); + return Arrays.stream((MergedAnnotation[]) repeated); + } + + private Stream> stream(MergedAnnotations annotations, + Class type) { + return annotations.stream(type.getName()) + .filter(MergedAnnotationPredicates.unique(MergedAnnotation::getMetaTypes)); + } + + private ConditionOutcome determineOutcome(MergedAnnotation annotation, PropertyResolver resolver) { + Class annotationType = annotation.getType(); + Spec spec = new Spec(annotationType, annotation.asAnnotationAttributes()); List missingProperties = new ArrayList<>(); List nonMatchingProperties = new ArrayList<>(); spec.collectProperties(resolver, missingProperties, nonMatchingProperties); if (!missingProperties.isEmpty()) { - return ConditionOutcome.noMatch(ConditionMessage.forCondition(ConditionalOnProperty.class, spec) + return ConditionOutcome.noMatch(ConditionMessage.forCondition(annotationType, spec) .didNotFind("property", "properties") .items(Style.QUOTE, missingProperties)); } if (!nonMatchingProperties.isEmpty()) { - return ConditionOutcome.noMatch(ConditionMessage.forCondition(ConditionalOnProperty.class, spec) + return ConditionOutcome.noMatch(ConditionMessage.forCondition(annotationType, spec) .found("different value in property", "different value in properties") .items(Style.QUOTE, nonMatchingProperties)); } - return ConditionOutcome - .match(ConditionMessage.forCondition(ConditionalOnProperty.class, spec).because("matched")); + return ConditionOutcome.match(ConditionMessage.forCondition(annotationType, spec).because("matched")); } private static class Spec { - private final String prefix; + private final Class annotationType; - private final String havingValue; + private final String prefix; private final String[] names; + private final String havingValue; + private final boolean matchIfMissing; - Spec(AnnotationAttributes annotationAttributes) { + Spec(Class annotationType, AnnotationAttributes annotationAttributes) { + this.annotationType = annotationType; + this.prefix = (!annotationAttributes.containsKey("prefix")) ? "" : getPrefix(annotationAttributes); + this.names = getNames(annotationAttributes); + this.havingValue = annotationAttributes.get("havingValue").toString(); + this.matchIfMissing = annotationAttributes.getBoolean("matchIfMissing"); + } + + private String getPrefix(AnnotationAttributes annotationAttributes) { String prefix = annotationAttributes.getString("prefix").trim(); if (StringUtils.hasText(prefix) && !prefix.endsWith(".")) { prefix = prefix + "."; } - this.prefix = prefix; - this.havingValue = annotationAttributes.getString("havingValue"); - this.names = getNames(annotationAttributes); - this.matchIfMissing = annotationAttributes.getBoolean("matchIfMissing"); + return prefix; } - private String[] getNames(Map annotationAttributes) { + private String[] getNames(AnnotationAttributes annotationAttributes) { String[] value = (String[]) annotationAttributes.get("value"); String[] name = (String[]) annotationAttributes.get("name"); Assert.state(value.length > 0 || name.length > 0, - "The name or value attribute of @ConditionalOnProperty must be specified"); + () -> "The name or value attribute of @%s must be specified" + .formatted(ClassUtils.getShortName(this.annotationType))); Assert.state(value.length == 0 || name.length == 0, - "The name and value attributes of @ConditionalOnProperty are exclusive"); + () -> "The name and value attributes of @%s are exclusive" + .formatted(ClassUtils.getShortName(this.annotationType))); return (value.length > 0) ? value : name; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnPropertyListCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnPropertyListCondition.java index 413de3b31fcf..1ab148c9f2b0 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnPropertyListCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnPropertyListCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * @author Stephane Nicoll * @since 2.0.5 */ -public class OnPropertyListCondition extends SpringBootCondition { +public abstract class OnPropertyListCondition extends SpringBootCondition { private static final Bindable> STRING_LIST = Bindable.listOf(String.class); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnResourceCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnResourceCondition.java index 16faa3ecb007..39ce2101cff2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnResourceCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnResourceCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,7 +45,7 @@ public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeM ResourceLoader loader = context.getResourceLoader(); List locations = new ArrayList<>(); collectValues(locations, attributes.get("resources")); - Assert.isTrue(!locations.isEmpty(), + Assert.state(!locations.isEmpty(), "@ConditionalOnResource annotations must specify at least one resource location"); List missing = new ArrayList<>(); for (String location : locations) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/context/LifecycleProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/context/LifecycleProperties.java index 49c706a8c223..5160581d78b1 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/context/LifecycleProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/context/LifecycleProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Andy Wilkinson * @since 2.3.0 */ -@ConfigurationProperties(prefix = "spring.lifecycle") +@ConfigurationProperties("spring.lifecycle") public class LifecycleProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/context/MessageSourceProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/context/MessageSourceProperties.java index 0495e285fd4f..e743f5809fdf 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/context/MessageSourceProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/context/MessageSourceProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ * @author Misagh Moayyed * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.messages") +@ConfigurationProperties("spring.messages") public class MessageSourceProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseAutoConfiguration.java index b935fbca3110..f6b54204d87c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -93,15 +93,16 @@ public class CouchbaseAutoConfiguration { @Bean @ConditionalOnMissingBean(CouchbaseConnectionDetails.class) - PropertiesCouchbaseConnectionDetails couchbaseConnectionDetails() { - return new PropertiesCouchbaseConnectionDetails(this.properties); + PropertiesCouchbaseConnectionDetails couchbaseConnectionDetails(ObjectProvider sslBundles) { + return new PropertiesCouchbaseConnectionDetails(this.properties, sslBundles.getIfAvailable()); } @Bean @ConditionalOnMissingBean public ClusterEnvironment couchbaseClusterEnvironment( - ObjectProvider customizers, ObjectProvider sslBundles) { - Builder builder = initializeEnvironmentBuilder(sslBundles.getIfAvailable()); + ObjectProvider customizers, + CouchbaseConnectionDetails connectionDetails) { + Builder builder = initializeEnvironmentBuilder(connectionDetails); customizers.orderedStream().forEach((customizer) -> customizer.customize(builder)); return builder.build(); } @@ -143,7 +144,7 @@ public Cluster couchbaseCluster(ClusterEnvironment couchbaseClusterEnvironment, return Cluster.connect(connectionDetails.getConnectionString(), options); } - private ClusterEnvironment.Builder initializeEnvironmentBuilder(SslBundles sslBundles) { + private ClusterEnvironment.Builder initializeEnvironmentBuilder(CouchbaseConnectionDetails connectionDetails) { ClusterEnvironment.Builder builder = ClusterEnvironment.builder(); Timeouts timeouts = this.properties.getEnv().getTimeouts(); builder.timeoutConfig((config) -> config.kvTimeout(timeouts.getKeyValue()) @@ -159,31 +160,24 @@ private ClusterEnvironment.Builder initializeEnvironmentBuilder(SslBundles sslBu builder.ioConfig((config) -> config.maxHttpConnections(io.getMaxEndpoints()) .numKvConnections(io.getMinEndpoints()) .idleHttpConnectionTimeout(io.getIdleHttpConnectionTimeout())); - if (this.properties.getEnv().getSsl().getEnabled()) { - configureSsl(builder, sslBundles); + SslBundle sslBundle = connectionDetails.getSslBundle(); + if (sslBundle != null) { + configureSsl(builder, sslBundle); } return builder; } - private void configureSsl(Builder builder, SslBundles sslBundles) { - Ssl sslProperties = this.properties.getEnv().getSsl(); - SslBundle sslBundle = (StringUtils.hasText(sslProperties.getBundle())) - ? sslBundles.getBundle(sslProperties.getBundle()) : null; - Assert.state(sslBundle == null || !sslBundle.getOptions().isSpecified(), - "SSL Options cannot be specified with Couchbase"); + private void configureSsl(Builder builder, SslBundle sslBundle) { + Assert.state(!sslBundle.getOptions().isSpecified(), "SSL Options cannot be specified with Couchbase"); builder.securityConfig((config) -> { config.enableTls(true); - TrustManagerFactory trustManagerFactory = getTrustManagerFactory(sslBundle); + TrustManagerFactory trustManagerFactory = sslBundle.getManagers().getTrustManagerFactory(); if (trustManagerFactory != null) { config.trustManagerFactory(trustManagerFactory); } }); } - private TrustManagerFactory getTrustManagerFactory(SslBundle sslBundle) { - return (sslBundle != null) ? sslBundle.getManagers().getTrustManagerFactory() : null; - } - @Configuration(proxyBeanMethods = false) @ConditionalOnClass(ObjectMapper.class) static class JacksonConfiguration { @@ -228,7 +222,7 @@ static final class CouchbaseCondition extends AnyNestedCondition { super(ConfigurationPhase.REGISTER_BEAN); } - @ConditionalOnProperty(prefix = "spring.couchbase", name = "connection-string") + @ConditionalOnProperty("spring.couchbase.connection-string") private static final class CouchbaseUrlCondition { } @@ -247,8 +241,11 @@ static final class PropertiesCouchbaseConnectionDetails implements CouchbaseConn private final CouchbaseProperties properties; - PropertiesCouchbaseConnectionDetails(CouchbaseProperties properties) { + private final SslBundles sslBundles; + + PropertiesCouchbaseConnectionDetails(CouchbaseProperties properties, SslBundles sslBundles) { this.properties = properties; + this.sslBundles = sslBundles; } @Override @@ -266,6 +263,19 @@ public String getPassword() { return this.properties.getPassword(); } + @Override + public SslBundle getSslBundle() { + Ssl ssl = this.properties.getEnv().getSsl(); + if (!ssl.getEnabled()) { + return null; + } + if (StringUtils.hasLength(ssl.getBundle())) { + Assert.notNull(this.sslBundles, "SSL bundle name has been set but no SSL bundles found in context"); + return this.sslBundles.getBundle(ssl.getBundle()); + } + return SslBundle.systemDefault(); + } + } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseConnectionDetails.java index f4ca1c76ee4b..8b5094b8e101 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,6 +17,7 @@ package org.springframework.boot.autoconfigure.couchbase; import org.springframework.boot.autoconfigure.service.connection.ConnectionDetails; +import org.springframework.boot.ssl.SslBundle; /** * Details required to establish a connection to a Couchbase service. @@ -46,4 +47,13 @@ public interface CouchbaseConnectionDetails extends ConnectionDetails { */ String getPassword(); + /** + * SSL bundle to use. + * @return the SSL bundle to use + * @since 3.5.0 + */ + default SslBundle getSslBundle() { + return null; + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseProperties.java index 2f4d2e8a0d05..42a091e4cc96 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ * @author Scott Frederick * @since 1.4.0 */ -@ConfigurationProperties(prefix = "spring.couchbase") +@ConfigurationProperties("spring.couchbase") public class CouchbaseProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/dao/PersistenceExceptionTranslationAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/dao/PersistenceExceptionTranslationAutoConfiguration.java index 2b48e77c3c6e..a5d79bcb6c28 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/dao/PersistenceExceptionTranslationAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/dao/PersistenceExceptionTranslationAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,9 +18,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Bean; import org.springframework.core.env.Environment; import org.springframework.dao.annotation.PersistenceExceptionTranslationPostProcessor; @@ -40,7 +40,7 @@ public class PersistenceExceptionTranslationAutoConfiguration { @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(prefix = "spring.dao.exceptiontranslation", name = "enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.dao.exceptiontranslation.enabled", matchIfMissing = true) public static PersistenceExceptionTranslationPostProcessor persistenceExceptionTranslationPostProcessor( Environment environment) { PersistenceExceptionTranslationPostProcessor postProcessor = new PersistenceExceptionTranslationPostProcessor(); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraDataAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraDataAutoConfiguration.java index 890b4895414e..f88a487bbedf 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraDataAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraDataAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,6 +45,8 @@ import org.springframework.data.cassandra.core.convert.CassandraConverter; import org.springframework.data.cassandra.core.convert.CassandraCustomConversions; import org.springframework.data.cassandra.core.convert.MappingCassandraConverter; +import org.springframework.data.cassandra.core.cql.CqlOperations; +import org.springframework.data.cassandra.core.cql.CqlTemplate; import org.springframework.data.cassandra.core.mapping.CassandraMappingContext; import org.springframework.data.cassandra.core.mapping.SimpleUserTypeResolver; @@ -114,10 +116,16 @@ public SessionFactoryFactoryBean cassandraSessionFactory(Environment environment return session; } + @Bean + @ConditionalOnMissingBean(CqlOperations.class) + public CqlTemplate cqlTemplate(SessionFactory sessionFactory) { + return new CqlTemplate(sessionFactory); + } + @Bean @ConditionalOnMissingBean(CassandraOperations.class) - public CassandraTemplate cassandraTemplate(SessionFactory sessionFactory, CassandraConverter converter) { - return new CassandraTemplate(sessionFactory, converter); + public CassandraTemplate cassandraTemplate(CqlTemplate cqlTemplate, CassandraConverter converter) { + return new CassandraTemplate(cqlTemplate, converter); } @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraReactiveDataAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraReactiveDataAutoConfiguration.java index 00d83b5f76da..911bd2cb11b2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraReactiveDataAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraReactiveDataAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,6 +30,8 @@ import org.springframework.data.cassandra.core.ReactiveCassandraOperations; import org.springframework.data.cassandra.core.ReactiveCassandraTemplate; import org.springframework.data.cassandra.core.convert.CassandraConverter; +import org.springframework.data.cassandra.core.cql.ReactiveCqlOperations; +import org.springframework.data.cassandra.core.cql.ReactiveCqlTemplate; import org.springframework.data.cassandra.core.cql.session.DefaultBridgedReactiveSession; import org.springframework.data.cassandra.core.cql.session.DefaultReactiveSessionFactory; @@ -58,11 +60,17 @@ public ReactiveSessionFactory reactiveCassandraSessionFactory(ReactiveSession re return new DefaultReactiveSessionFactory(reactiveCassandraSession); } + @Bean + @ConditionalOnMissingBean(ReactiveCqlOperations.class) + public ReactiveCqlTemplate reactiveCqlTemplate(ReactiveSessionFactory reactiveCassandraSessionFactory) { + return new ReactiveCqlTemplate(reactiveCassandraSessionFactory); + } + @Bean @ConditionalOnMissingBean(ReactiveCassandraOperations.class) - public ReactiveCassandraTemplate reactiveCassandraTemplate(ReactiveSession reactiveCassandraSession, + public ReactiveCassandraTemplate reactiveCassandraTemplate(ReactiveCqlTemplate reactiveCqlTemplate, CassandraConverter converter) { - return new ReactiveCassandraTemplate(reactiveCassandraSession, converter); + return new ReactiveCassandraTemplate(reactiveCqlTemplate, converter); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/couchbase/CouchbaseDataProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/couchbase/CouchbaseDataProperties.java index 359beeafaaca..694eebd3ec7e 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/couchbase/CouchbaseDataProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/couchbase/CouchbaseDataProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,7 @@ * @author Stephane Nicoll * @since 1.4.0 */ -@ConfigurationProperties(prefix = "spring.data.couchbase") +@ConfigurationProperties("spring.data.couchbase") public class CouchbaseDataProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/elasticsearch/ElasticsearchRepositoriesAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/elasticsearch/ElasticsearchRepositoriesAutoConfiguration.java index ed6a7fe2298f..2568ad417bbf 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/elasticsearch/ElasticsearchRepositoriesAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/elasticsearch/ElasticsearchRepositoriesAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,9 +18,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Import; import org.springframework.data.elasticsearch.repository.ElasticsearchRepository; import org.springframework.data.elasticsearch.repository.config.EnableElasticsearchRepositories; @@ -37,8 +37,7 @@ */ @AutoConfiguration @ConditionalOnClass(ElasticsearchRepository.class) -@ConditionalOnProperty(prefix = "spring.data.elasticsearch.repositories", name = "enabled", havingValue = "true", - matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.data.elasticsearch.repositories.enabled", matchIfMissing = true) @ConditionalOnMissingBean(ElasticsearchRepositoryFactoryBean.class) @Import(ElasticsearchRepositoriesRegistrar.class) public class ElasticsearchRepositoriesAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/elasticsearch/ReactiveElasticsearchRepositoriesAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/elasticsearch/ReactiveElasticsearchRepositoriesAutoConfiguration.java index cdf64537a61d..85f3b316f299 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/elasticsearch/ReactiveElasticsearchRepositoriesAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/elasticsearch/ReactiveElasticsearchRepositoriesAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,9 +20,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Import; import org.springframework.data.elasticsearch.client.elc.ReactiveElasticsearchClient; import org.springframework.data.elasticsearch.repository.ReactiveElasticsearchRepository; @@ -39,8 +39,7 @@ */ @AutoConfiguration @ConditionalOnClass({ ReactiveElasticsearchClient.class, ReactiveElasticsearchRepository.class, Mono.class }) -@ConditionalOnProperty(prefix = "spring.data.elasticsearch.repositories", name = "enabled", havingValue = "true", - matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.data.elasticsearch.repositories.enabled", matchIfMissing = true) @ConditionalOnMissingBean(ReactiveElasticsearchRepositoryFactoryBean.class) @Import(ReactiveElasticsearchRepositoriesRegistrar.class) public class ReactiveElasticsearchRepositoriesAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jdbc/JdbcDataProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jdbc/JdbcDataProperties.java index bc2f8a95b71c..2105f1515f15 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jdbc/JdbcDataProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jdbc/JdbcDataProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2024-2024 the original author or authors. + * Copyright 2024-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,7 @@ * @author Jens Schauder * @since 3.3.0 */ -@ConfigurationProperties(prefix = "spring.data.jdbc") +@ConfigurationProperties("spring.data.jdbc") public class JdbcDataProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jdbc/JdbcRepositoriesAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jdbc/JdbcRepositoriesAutoConfiguration.java index 4ba437be7b19..37d0be41e281 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jdbc/JdbcRepositoriesAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jdbc/JdbcRepositoriesAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,9 +22,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.domain.EntityScanner; import org.springframework.boot.autoconfigure.jdbc.DataSourceTransactionManagerAutoConfiguration; import org.springframework.boot.autoconfigure.jdbc.JdbcTemplateAutoConfiguration; @@ -67,8 +67,7 @@ @AutoConfiguration(after = { JdbcTemplateAutoConfiguration.class, DataSourceTransactionManagerAutoConfiguration.class }) @ConditionalOnBean({ NamedParameterJdbcOperations.class, PlatformTransactionManager.class }) @ConditionalOnClass({ NamedParameterJdbcOperations.class, AbstractJdbcConfiguration.class }) -@ConditionalOnProperty(prefix = "spring.data.jdbc.repositories", name = "enabled", havingValue = "true", - matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.data.jdbc.repositories.enabled", matchIfMissing = true) @EnableConfigurationProperties(JdbcDataProperties.class) public class JdbcRepositoriesAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jpa/JpaRepositoriesAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jpa/JpaRepositoriesAutoConfiguration.java index 625598c109a5..08cf97b59dc6 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jpa/JpaRepositoriesAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/jpa/JpaRepositoriesAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,6 +24,7 @@ import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.AnyNestedCondition; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; @@ -74,8 +75,7 @@ @ConditionalOnBean(DataSource.class) @ConditionalOnClass(JpaRepository.class) @ConditionalOnMissingBean({ JpaRepositoryFactoryBean.class, JpaRepositoryConfigExtension.class }) -@ConditionalOnProperty(prefix = "spring.data.jpa.repositories", name = "enabled", havingValue = "true", - matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.data.jpa.repositories.enabled", matchIfMissing = true) @Import(JpaRepositoriesImportSelector.class) public class JpaRepositoriesAutoConfiguration { @@ -104,13 +104,12 @@ private static final class BootstrapExecutorCondition extends AnyNestedCondition super(ConfigurationPhase.REGISTER_BEAN); } - @ConditionalOnProperty(prefix = "spring.data.jpa.repositories", name = "bootstrap-mode", - havingValue = "deferred") + @ConditionalOnProperty(name = "spring.data.jpa.repositories.bootstrap-mode", havingValue = "deferred") static class DeferredBootstrapMode { } - @ConditionalOnProperty(prefix = "spring.data.jpa.repositories", name = "bootstrap-mode", havingValue = "lazy") + @ConditionalOnProperty(name = "spring.data.jpa.repositories.bootstrap-mode", havingValue = "lazy") static class LazyBootstrapMode { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/ldap/LdapRepositoriesAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/ldap/LdapRepositoriesAutoConfiguration.java index 4dc1795c6058..c00870c87eba 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/ldap/LdapRepositoriesAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/ldap/LdapRepositoriesAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,9 +20,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Import; import org.springframework.data.ldap.repository.LdapRepository; import org.springframework.data.ldap.repository.support.LdapRepositoryFactoryBean; @@ -35,8 +35,7 @@ */ @AutoConfiguration @ConditionalOnClass({ LdapContext.class, LdapRepository.class }) -@ConditionalOnProperty(prefix = "spring.data.ldap.repositories", name = "enabled", havingValue = "true", - matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.data.ldap.repositories.enabled", matchIfMissing = true) @ConditionalOnMissingBean(LdapRepositoryFactoryBean.class) @Import(LdapRepositoriesRegistrar.class) public class LdapRepositoriesAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/mongo/MongoDataAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/mongo/MongoDataAutoConfiguration.java index db1ac8876ed7..b14f931f52e3 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/mongo/MongoDataAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/mongo/MongoDataAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,6 +18,7 @@ import com.mongodb.client.MongoClient; +import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; @@ -27,6 +28,7 @@ import org.springframework.boot.autoconfigure.mongo.MongoProperties; import org.springframework.boot.autoconfigure.mongo.PropertiesMongoConnectionDetails; import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.boot.ssl.SslBundles; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Import; import org.springframework.data.mongodb.core.MongoTemplate; @@ -59,8 +61,9 @@ public class MongoDataAutoConfiguration { @Bean @ConditionalOnMissingBean(MongoConnectionDetails.class) - PropertiesMongoConnectionDetails mongoConnectionDetails(MongoProperties properties) { - return new PropertiesMongoConnectionDetails(properties); + PropertiesMongoConnectionDetails mongoConnectionDetails(MongoProperties properties, + ObjectProvider sslBundles) { + return new PropertiesMongoConnectionDetails(properties, sslBundles.getIfAvailable()); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/mongo/MongoDatabaseFactoryDependentConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/mongo/MongoDatabaseFactoryDependentConfiguration.java index ba0f6a1a95f0..381db02c36f5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/mongo/MongoDatabaseFactoryDependentConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/mongo/MongoDatabaseFactoryDependentConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -77,8 +77,8 @@ static class GridFsMongoDatabaseFactory implements MongoDatabaseFactory { GridFsMongoDatabaseFactory(MongoDatabaseFactory mongoDatabaseFactory, MongoConnectionDetails connectionDetails) { - Assert.notNull(mongoDatabaseFactory, "MongoDatabaseFactory must not be null"); - Assert.notNull(connectionDetails, "ConnectionDetails must not be null"); + Assert.notNull(mongoDatabaseFactory, "'mongoDatabaseFactory' must not be null"); + Assert.notNull(connectionDetails, "'connectionDetails' must not be null"); this.mongoDatabaseFactory = mongoDatabaseFactory; this.connectionDetails = connectionDetails; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/neo4j/Neo4jDataProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/neo4j/Neo4jDataProperties.java index ea1c19052472..100708347eaa 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/neo4j/Neo4jDataProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/neo4j/Neo4jDataProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,7 @@ * @author Michael J. Simons * @since 2.4.0 */ -@ConfigurationProperties(prefix = "spring.data.neo4j") +@ConfigurationProperties("spring.data.neo4j") public class Neo4jDataProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/r2dbc/R2dbcRepositoriesAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/r2dbc/R2dbcRepositoriesAutoConfiguration.java index a3a0daa6a8af..58b82c0e5cd9 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/r2dbc/R2dbcRepositoriesAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/r2dbc/R2dbcRepositoriesAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,9 +21,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Import; import org.springframework.data.r2dbc.repository.R2dbcRepository; import org.springframework.data.r2dbc.repository.config.EnableR2dbcRepositories; @@ -40,8 +40,7 @@ @AutoConfiguration(after = R2dbcDataAutoConfiguration.class) @ConditionalOnClass({ ConnectionFactory.class, R2dbcRepository.class }) @ConditionalOnBean(DatabaseClient.class) -@ConditionalOnProperty(prefix = "spring.data.r2dbc.repositories", name = "enabled", havingValue = "true", - matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.data.r2dbc.repositories.enabled", matchIfMissing = true) @ConditionalOnMissingBean(R2dbcRepositoryFactoryBean.class) @Import(R2dbcRepositoriesAutoConfigureRegistrar.class) public class R2dbcRepositoriesAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/JedisConnectionConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/JedisConnectionConfiguration.java index c1c6fbb6e811..353b2ef9c9db 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/JedisConnectionConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/JedisConnectionConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -92,21 +92,17 @@ JedisConnectionFactory redisConnectionFactoryVirtualThreads( private JedisConnectionFactory createJedisConnectionFactory( ObjectProvider builderCustomizers) { JedisClientConfiguration clientConfiguration = getJedisClientConfiguration(builderCustomizers); - if (getSentinelConfig() != null) { - return new JedisConnectionFactory(getSentinelConfig(), clientConfiguration); - } - if (getClusterConfiguration() != null) { - return new JedisConnectionFactory(getClusterConfiguration(), clientConfiguration); - } - return new JedisConnectionFactory(getStandaloneConfig(), clientConfiguration); + return switch (this.mode) { + case STANDALONE -> new JedisConnectionFactory(getStandaloneConfig(), clientConfiguration); + case CLUSTER -> new JedisConnectionFactory(getClusterConfiguration(), clientConfiguration); + case SENTINEL -> new JedisConnectionFactory(getSentinelConfig(), clientConfiguration); + }; } private JedisClientConfiguration getJedisClientConfiguration( ObjectProvider builderCustomizers) { JedisClientConfigurationBuilder builder = applyProperties(JedisClientConfiguration.builder()); - if (isSslEnabled()) { - applySsl(builder); - } + applySslIfNeeded(builder); RedisProperties.Pool pool = getProperties().getJedis().getPool(); if (isPoolEnabled(pool)) { applyPooling(pool, builder); @@ -126,18 +122,19 @@ private JedisClientConfigurationBuilder applyProperties(JedisClientConfiguration return builder; } - private void applySsl(JedisClientConfigurationBuilder builder) { - JedisSslClientConfigurationBuilder sslBuilder = builder.useSsl(); - if (getProperties().getSsl().getBundle() != null) { - SslBundle sslBundle = getSslBundles().getBundle(getProperties().getSsl().getBundle()); - sslBuilder.sslSocketFactory(sslBundle.createSslContext().getSocketFactory()); - SslOptions sslOptions = sslBundle.getOptions(); - SSLParameters sslParameters = new SSLParameters(); - PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull(); - map.from(sslOptions.getCiphers()).to(sslParameters::setCipherSuites); - map.from(sslOptions.getEnabledProtocols()).to(sslParameters::setProtocols); - sslBuilder.sslParameters(sslParameters); + private void applySslIfNeeded(JedisClientConfigurationBuilder builder) { + SslBundle sslBundle = getSslBundle(); + if (sslBundle == null) { + return; } + JedisSslClientConfigurationBuilder sslBuilder = builder.useSsl(); + sslBuilder.sslSocketFactory(sslBundle.createSslContext().getSocketFactory()); + SslOptions sslOptions = sslBundle.getOptions(); + SSLParameters sslParameters = new SSLParameters(); + PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull(); + map.from(sslOptions.getCiphers()).to(sslParameters::setCipherSuites); + map.from(sslOptions.getEnabledProtocols()).to(sslParameters::setProtocols); + sslBuilder.sslParameters(sslParameters); } private void applyPooling(RedisProperties.Pool pool, diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/LettuceConnectionConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/LettuceConnectionConfiguration.java index 040eebd919cd..734c747e51b5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/LettuceConnectionConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/LettuceConnectionConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,9 +19,11 @@ import java.time.Duration; import io.lettuce.core.ClientOptions; +import io.lettuce.core.ReadFrom; import io.lettuce.core.RedisClient; import io.lettuce.core.SocketOptions; import io.lettuce.core.TimeoutOptions; +import io.lettuce.core.api.StatefulConnection; import io.lettuce.core.cluster.ClusterClientOptions; import io.lettuce.core.cluster.ClusterTopologyRefreshOptions; import io.lettuce.core.cluster.ClusterTopologyRefreshOptions.Builder; @@ -114,19 +116,14 @@ private LettuceConnectionFactory createConnectionFactory( ObjectProvider clientConfigurationBuilderCustomizers, ObjectProvider clientOptionsBuilderCustomizers, ClientResources clientResources) { - LettuceClientConfiguration clientConfig = getLettuceClientConfiguration(clientConfigurationBuilderCustomizers, - clientOptionsBuilderCustomizers, clientResources, getProperties().getLettuce().getPool()); - return createLettuceConnectionFactory(clientConfig); - } - - private LettuceConnectionFactory createLettuceConnectionFactory(LettuceClientConfiguration clientConfiguration) { - if (getSentinelConfig() != null) { - return new LettuceConnectionFactory(getSentinelConfig(), clientConfiguration); - } - if (getClusterConfiguration() != null) { - return new LettuceConnectionFactory(getClusterConfiguration(), clientConfiguration); - } - return new LettuceConnectionFactory(getStandaloneConfig(), clientConfiguration); + LettuceClientConfiguration clientConfiguration = getLettuceClientConfiguration( + clientConfigurationBuilderCustomizers, clientOptionsBuilderCustomizers, clientResources, + getProperties().getLettuce().getPool()); + return switch (this.mode) { + case STANDALONE -> new LettuceConnectionFactory(getStandaloneConfig(), clientConfiguration); + case CLUSTER -> new LettuceConnectionFactory(getClusterConfiguration(), clientConfiguration); + case SENTINEL -> new LettuceConnectionFactory(getSentinelConfig(), clientConfiguration); + }; } private LettuceClientConfiguration getLettuceClientConfiguration( @@ -134,11 +131,12 @@ private LettuceClientConfiguration getLettuceClientConfiguration( ObjectProvider clientOptionsBuilderCustomizers, ClientResources clientResources, Pool pool) { LettuceClientConfigurationBuilder builder = createBuilder(pool); - applyProperties(builder); + SslBundle sslBundle = getSslBundle(); + applyProperties(builder, sslBundle); if (StringUtils.hasText(getProperties().getUrl())) { customizeConfigurationFromUrl(builder); } - builder.clientOptions(createClientOptions(clientOptionsBuilderCustomizers)); + builder.clientOptions(createClientOptions(clientOptionsBuilderCustomizers, sslBundle)); builder.clientResources(clientResources); clientConfigurationBuilderCustomizers.orderedStream().forEach((customizer) -> customizer.customize(builder)); return builder.build(); @@ -151,8 +149,8 @@ private LettuceClientConfigurationBuilder createBuilder(Pool pool) { return LettuceClientConfiguration.builder(); } - private void applyProperties(LettuceClientConfiguration.LettuceClientConfigurationBuilder builder) { - if (isSslEnabled()) { + private void applyProperties(LettuceClientConfigurationBuilder builder, SslBundle sslBundle) { + if (sslBundle != null) { builder.useSsl(); } if (getProperties().getTimeout() != null) { @@ -163,21 +161,44 @@ private void applyProperties(LettuceClientConfiguration.LettuceClientConfigurati if (lettuce.getShutdownTimeout() != null && !lettuce.getShutdownTimeout().isZero()) { builder.shutdownTimeout(getProperties().getLettuce().getShutdownTimeout()); } + String readFrom = lettuce.getReadFrom(); + if (readFrom != null) { + builder.readFrom(getReadFrom(readFrom)); + } } if (StringUtils.hasText(getProperties().getClientName())) { builder.clientName(getProperties().getClientName()); } } + private ReadFrom getReadFrom(String readFrom) { + int index = readFrom.indexOf(':'); + if (index == -1) { + return ReadFrom.valueOf(getCanonicalReadFromName(readFrom)); + } + String name = getCanonicalReadFromName(readFrom.substring(0, index)); + String value = readFrom.substring(index + 1); + return ReadFrom.valueOf(name + ":" + value); + } + + private String getCanonicalReadFromName(String name) { + StringBuilder canonicalName = new StringBuilder(name.length()); + name.chars() + .filter(Character::isLetterOrDigit) + .map(Character::toLowerCase) + .forEach((c) -> canonicalName.append((char) c)); + return canonicalName.toString(); + } + private ClientOptions createClientOptions( - ObjectProvider clientConfigurationBuilderCustomizers) { + ObjectProvider clientConfigurationBuilderCustomizers, + SslBundle sslBundle) { ClientOptions.Builder builder = initializeClientOptionsBuilder(); Duration connectTimeout = getProperties().getConnectTimeout(); if (connectTimeout != null) { builder.socketOptions(SocketOptions.builder().connectTimeout(connectTimeout).build()); } - if (isSslEnabled() && getProperties().getSsl().getBundle() != null) { - SslBundle sslBundle = getSslBundles().getBundle(getProperties().getSsl().getBundle()); + if (sslBundle != null) { io.lettuce.core.SslOptions.Builder sslOptionsBuilder = io.lettuce.core.SslOptions.builder(); sslOptionsBuilder.keyManager(sslBundle.getManagers().getKeyManagerFactory()); sslOptionsBuilder.trustManager(sslBundle.getManagers().getTrustManagerFactory()); @@ -227,8 +248,8 @@ LettuceClientConfigurationBuilder createBuilder(Pool properties) { return LettucePoolingClientConfiguration.builder().poolConfig(getPoolConfig(properties)); } - private GenericObjectPoolConfig getPoolConfig(Pool properties) { - GenericObjectPoolConfig config = new GenericObjectPoolConfig<>(); + private GenericObjectPoolConfig> getPoolConfig(Pool properties) { + GenericObjectPoolConfig> config = new GenericObjectPoolConfig<>(); config.setMaxTotal(properties.getMaxActive()); config.setMaxIdle(properties.getMaxIdle()); config.setMinIdle(properties.getMinIdle()); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/PropertiesRedisConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/PropertiesRedisConnectionDetails.java index 18371b3200fd..7b42223c6914 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/PropertiesRedisConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/PropertiesRedisConnectionDetails.java @@ -18,7 +18,10 @@ import java.util.List; -import org.springframework.boot.autoconfigure.data.redis.RedisConnectionConfiguration.ConnectionInfo; +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslBundles; +import org.springframework.util.Assert; +import org.springframework.util.StringUtils; /** * Adapts {@link RedisProperties} to {@link RedisConnectionDetails}. @@ -27,85 +30,71 @@ * @author Andy Wilkinson * @author Phillip Webb * @author Scott Frederick + * @author Yanming Zhou + * @author Phillip Webb */ class PropertiesRedisConnectionDetails implements RedisConnectionDetails { private final RedisProperties properties; - PropertiesRedisConnectionDetails(RedisProperties properties) { + private final SslBundles sslBundles; + + PropertiesRedisConnectionDetails(RedisProperties properties, SslBundles sslBundles) { this.properties = properties; + this.sslBundles = sslBundles; } @Override public String getUsername() { - if (this.properties.getUrl() != null) { - ConnectionInfo connectionInfo = ConnectionInfo.of(this.properties.getUrl()); - return connectionInfo.getUsername(); - } - return this.properties.getUsername(); + RedisUrl redisUrl = getRedisUrl(); + return (redisUrl != null) ? redisUrl.credentials().username() : this.properties.getUsername(); } @Override public String getPassword() { - if (this.properties.getUrl() != null) { - ConnectionInfo connectionInfo = ConnectionInfo.of(this.properties.getUrl()); - return connectionInfo.getPassword(); - } - return this.properties.getPassword(); + RedisUrl redisUrl = getRedisUrl(); + return (redisUrl != null) ? redisUrl.credentials().password() : this.properties.getPassword(); } @Override public Standalone getStandalone() { - if (this.properties.getUrl() != null) { - ConnectionInfo connectionInfo = ConnectionInfo.of(this.properties.getUrl()); - return Standalone.of(connectionInfo.getUri().getHost(), connectionInfo.getUri().getPort(), - this.properties.getDatabase()); + RedisUrl redisUrl = getRedisUrl(); + return (redisUrl != null) + ? Standalone.of(redisUrl.uri().getHost(), redisUrl.uri().getPort(), redisUrl.database(), getSslBundle()) + : Standalone.of(this.properties.getHost(), this.properties.getPort(), this.properties.getDatabase(), + getSslBundle()); + } + + private SslBundle getSslBundle() { + if (!this.properties.getSsl().isEnabled()) { + return null; + } + String bundleName = this.properties.getSsl().getBundle(); + if (StringUtils.hasLength(bundleName)) { + Assert.notNull(this.sslBundles, "SSL bundle name has been set but no SSL bundles found in context"); + return this.sslBundles.getBundle(bundleName); } - return Standalone.of(this.properties.getHost(), this.properties.getPort(), this.properties.getDatabase()); + return SslBundle.systemDefault(); } @Override public Sentinel getSentinel() { - org.springframework.boot.autoconfigure.data.redis.RedisProperties.Sentinel sentinel = this.properties - .getSentinel(); - if (sentinel == null) { - return null; - } - return new Sentinel() { - - @Override - public int getDatabase() { - return PropertiesRedisConnectionDetails.this.properties.getDatabase(); - } - - @Override - public String getMaster() { - return sentinel.getMaster(); - } - - @Override - public List getNodes() { - return sentinel.getNodes().stream().map(PropertiesRedisConnectionDetails.this::asNode).toList(); - } - - @Override - public String getUsername() { - return sentinel.getUsername(); - } - - @Override - public String getPassword() { - return sentinel.getPassword(); - } - - }; + RedisProperties.Sentinel sentinel = this.properties.getSentinel(); + return (sentinel != null) ? new PropertiesSentinel(getStandalone().getDatabase(), sentinel) : null; } @Override public Cluster getCluster() { RedisProperties.Cluster cluster = this.properties.getCluster(); - List nodes = (cluster != null) ? cluster.getNodes().stream().map(this::asNode).toList() : null; - return (nodes != null) ? () -> nodes : null; + return (cluster != null) ? new PropertiesCluster(cluster) : null; + } + + private RedisUrl getRedisUrl() { + return RedisUrl.of(this.properties.getUrl()); + } + + private List asNodes(List nodes) { + return nodes.stream().map(this::asNode).toList(); } private Node asNode(String node) { @@ -115,4 +104,73 @@ private Node asNode(String node) { return new Node(host, port); } + /** + * {@link Cluster} implementation backed by properties. + */ + private class PropertiesCluster implements Cluster { + + private final List nodes; + + PropertiesCluster(RedisProperties.Cluster properties) { + this.nodes = asNodes(properties.getNodes()); + } + + @Override + public List getNodes() { + return this.nodes; + } + + @Override + public SslBundle getSslBundle() { + return PropertiesRedisConnectionDetails.this.getSslBundle(); + } + + } + + /** + * {@link Sentinel} implementation backed by properties. + */ + private class PropertiesSentinel implements Sentinel { + + private final int database; + + private final RedisProperties.Sentinel properties; + + PropertiesSentinel(int database, RedisProperties.Sentinel properties) { + this.database = database; + this.properties = properties; + } + + @Override + public int getDatabase() { + return this.database; + } + + @Override + public String getMaster() { + return this.properties.getMaster(); + } + + @Override + public List getNodes() { + return asNodes(this.properties.getNodes()); + } + + @Override + public String getUsername() { + return this.properties.getUsername(); + } + + @Override + public String getPassword() { + return this.properties.getPassword(); + } + + @Override + public SslBundle getSslBundle() { + return PropertiesRedisConnectionDetails.this.getSslBundle(); + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisAutoConfiguration.java index fdef9695ff5e..c5480cc9a96f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,12 +16,14 @@ package org.springframework.boot.autoconfigure.data.redis; +import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnSingleCandidate; import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.boot.ssl.SslBundles; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Import; import org.springframework.data.redis.connection.RedisConnectionFactory; @@ -51,8 +53,9 @@ public class RedisAutoConfiguration { @Bean @ConditionalOnMissingBean(RedisConnectionDetails.class) - PropertiesRedisConnectionDetails redisConnectionDetails(RedisProperties properties) { - return new PropertiesRedisConnectionDetails(properties); + PropertiesRedisConnectionDetails redisConnectionDetails(RedisProperties properties, + ObjectProvider sslBundles) { + return new PropertiesRedisConnectionDetails(properties, sslBundles.getIfAvailable()); } @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisConnectionConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisConnectionConfiguration.java index 4b19d54550b4..d3a703608eea 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisConnectionConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisConnectionConfiguration.java @@ -16,8 +16,6 @@ package org.springframework.boot.autoconfigure.data.redis; -import java.net.URI; -import java.net.URISyntaxException; import java.util.ArrayList; import java.util.List; @@ -26,6 +24,7 @@ import org.springframework.boot.autoconfigure.data.redis.RedisConnectionDetails.Node; import org.springframework.boot.autoconfigure.data.redis.RedisConnectionDetails.Sentinel; import org.springframework.boot.autoconfigure.data.redis.RedisProperties.Pool; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.ssl.SslBundles; import org.springframework.data.redis.connection.RedisClusterConfiguration; import org.springframework.data.redis.connection.RedisNode; @@ -45,6 +44,7 @@ * @author Moritz Halbritter * @author Andy Wilkinson * @author Phillip Webb + * @author Yanming Zhou */ abstract class RedisConnectionConfiguration { @@ -63,6 +63,8 @@ abstract class RedisConnectionConfiguration { private final SslBundles sslBundles; + protected final Mode mode; + protected RedisConnectionConfiguration(RedisProperties properties, RedisConnectionDetails connectionDetails, ObjectProvider standaloneConfigurationProvider, ObjectProvider sentinelConfigurationProvider, @@ -74,6 +76,7 @@ protected RedisConnectionConfiguration(RedisProperties properties, RedisConnecti this.clusterConfiguration = clusterConfigurationProvider.getIfAvailable(); this.connectionDetails = connectionDetails; this.sslBundles = sslBundles.getIfAvailable(); + this.mode = determineMode(); } protected final RedisStandaloneConfiguration getStandaloneConfig() { @@ -150,14 +153,29 @@ protected final RedisProperties getProperties() { return this.properties; } - protected SslBundles getSslBundles() { + protected final SslBundles getSslBundles() { return this.sslBundles; } - protected boolean isSslEnabled() { + protected SslBundle getSslBundle() { + return switch (this.mode) { + case STANDALONE -> (this.connectionDetails.getStandalone() != null) + ? this.connectionDetails.getStandalone().getSslBundle() : null; + case CLUSTER -> (this.connectionDetails.getCluster() != null) + ? this.connectionDetails.getCluster().getSslBundle() : null; + case SENTINEL -> (this.connectionDetails.getSentinel() != null) + ? this.connectionDetails.getSentinel().getSslBundle() : null; + }; + } + + protected final boolean isSslEnabled() { return getProperties().getSsl().isEnabled(); } + protected final boolean urlUsesSsl() { + return RedisUrl.of(this.properties.getUrl()).useSsl(); + } + protected boolean isPoolEnabled(Pool pool) { Boolean enabled = pool.getEnabled(); return (enabled != null) ? enabled : COMMONS_POOL2_AVAILABLE; @@ -171,74 +189,23 @@ private List createSentinels(Sentinel sentinel) { return nodes; } - protected final boolean urlUsesSsl() { - return ConnectionInfo.of(this.properties.getUrl()).isUseSsl(); - } - protected final RedisConnectionDetails getConnectionDetails() { return this.connectionDetails; } - static final class ConnectionInfo { - - private final URI uri; - - private final boolean useSsl; - - private final String username; - - private final String password; - - private ConnectionInfo(URI uri, boolean useSsl, String username, String password) { - this.uri = uri; - this.useSsl = useSsl; - this.username = username; - this.password = password; - } - - URI getUri() { - return this.uri; - } - - boolean isUseSsl() { - return this.useSsl; + private Mode determineMode() { + if (getSentinelConfig() != null) { + return Mode.SENTINEL; } - - String getUsername() { - return this.username; + if (getClusterConfiguration() != null) { + return Mode.CLUSTER; } + return Mode.STANDALONE; + } - String getPassword() { - return this.password; - } + enum Mode { - static ConnectionInfo of(String url) { - try { - URI uri = new URI(url); - String scheme = uri.getScheme(); - if (!"redis".equals(scheme) && !"rediss".equals(scheme)) { - throw new RedisUrlSyntaxException(url); - } - boolean useSsl = ("rediss".equals(scheme)); - String username = null; - String password = null; - if (uri.getUserInfo() != null) { - String candidate = uri.getUserInfo(); - int index = candidate.indexOf(':'); - if (index >= 0) { - username = candidate.substring(0, index); - password = candidate.substring(index + 1); - } - else { - password = candidate; - } - } - return new ConnectionInfo(uri, useSsl, username, password); - } - catch (URISyntaxException ex) { - throw new RedisUrlSyntaxException(url, ex); - } - } + STANDALONE, CLUSTER, SENTINEL } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisConnectionDetails.java index b423e61ab2eb..013aaee5d0b1 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,7 @@ import java.util.List; import org.springframework.boot.autoconfigure.service.connection.ConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.util.Assert; /** @@ -98,12 +99,59 @@ default int getDatabase() { return 0; } + /** + * SSL bundle to use. + * @return the SSL bundle to use + * @since 3.5.0 + */ + default SslBundle getSslBundle() { + return null; + } + + /** + * Creates a new instance with the given host and port. + * @param host the host + * @param port the port + * @return the new instance + */ static Standalone of(String host, int port) { - return of(host, port, 0); + return of(host, port, 0, null); } + /** + * Creates a new instance with the given host, port and SSL bundle. + * @param host the host + * @param port the port + * @param sslBundle the SSL bundle + * @return the new instance + * @since 3.5.0 + */ + static Standalone of(String host, int port, SslBundle sslBundle) { + return of(host, port, 0, sslBundle); + } + + /** + * Creates a new instance with the given host, port and database. + * @param host the host + * @param port the port + * @param database the database + * @return the new instance + */ static Standalone of(String host, int port, int database) { - Assert.hasLength(host, "Host must not be empty"); + return of(host, port, database, null); + } + + /** + * Creates a new instance with the given host, port, database and SSL bundle. + * @param host the host + * @param port the port + * @param database the database + * @param sslBundle the SSL bundle + * @return the new instance + * @since 3.5.0 + */ + static Standalone of(String host, int port, int database, SslBundle sslBundle) { + Assert.hasLength(host, "'host' must not be empty"); return new Standalone() { @Override @@ -121,6 +169,10 @@ public int getDatabase() { return database; } + @Override + public SslBundle getSslBundle() { + return sslBundle; + } }; } @@ -161,6 +213,15 @@ interface Sentinel { */ String getPassword(); + /** + * SSL bundle to use. + * @return the SSL bundle to use + * @since 3.5.0 + */ + default SslBundle getSslBundle() { + return null; + } + } /** @@ -175,6 +236,15 @@ interface Cluster { */ List getNodes(); + /** + * SSL bundle to use. + * @return the SSL bundle to use + * @since 3.5.0 + */ + default SslBundle getSslBundle() { + return null; + } + } /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisProperties.java index 8e5f07eb5bf3..ca7a85875ab4 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,9 +31,10 @@ * @author Mark Paluch * @author Stephane Nicoll * @author Scott Frederick + * @author Yanming Zhou * @since 1.0.0 */ -@ConfigurationProperties(prefix = "spring.data.redis") +@ConfigurationProperties("spring.data.redis") public class RedisProperties { /** @@ -42,8 +43,8 @@ public class RedisProperties { private int database = 0; /** - * Connection URL. Overrides host, port, username, and password. Example: - * redis://user:password@example.com:6379 + * Connection URL. Overrides host, port, username, password, and database. Example: + * redis://user:password@example.com:6379/8 */ private String url; @@ -467,6 +468,11 @@ public static class Lettuce { */ private Duration shutdownTimeout = Duration.ofMillis(100); + /** + * Defines from which Redis nodes data is read. + */ + private String readFrom; + /** * Lettuce pool configuration. */ @@ -482,6 +488,14 @@ public void setShutdownTimeout(Duration shutdownTimeout) { this.shutdownTimeout = shutdownTimeout; } + public void setReadFrom(String readFrom) { + this.readFrom = readFrom; + } + + public String getReadFrom() { + return this.readFrom; + } + public Pool getPool() { return this.pool; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisRepositoriesAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisRepositoriesAutoConfiguration.java index 1c8e5ceca2a1..442247b85ed2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisRepositoriesAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisRepositoriesAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,9 +19,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Import; import org.springframework.data.redis.connection.RedisConnectionFactory; import org.springframework.data.redis.repository.configuration.EnableRedisRepositories; @@ -39,8 +39,7 @@ @AutoConfiguration(after = RedisAutoConfiguration.class) @ConditionalOnClass(EnableRedisRepositories.class) @ConditionalOnBean(RedisConnectionFactory.class) -@ConditionalOnProperty(prefix = "spring.data.redis.repositories", name = "enabled", havingValue = "true", - matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.data.redis.repositories.enabled", matchIfMissing = true) @ConditionalOnMissingBean(RedisRepositoryFactoryBean.class) @Import(RedisRepositoriesRegistrar.class) public class RedisRepositoriesAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisUrl.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisUrl.java new file mode 100644 index 000000000000..7639b16e77fe --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/redis/RedisUrl.java @@ -0,0 +1,98 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.data.redis; + +import java.net.URI; +import java.net.URISyntaxException; + +import org.springframework.util.StringUtils; + +/** + * A parsed URL used to connect to Redis. + * + * @param uri the source URI + * @param useSsl if SSL is used to connect + * @param credentials the connection credentials + * @param database the database index + * @author Mark Paluch + * @author Stephane Nicoll + * @author Alen Turkovic + * @author Scott Frederick + * @author Eddú Meléndez + * @author Moritz Halbritter + * @author Andy Wilkinson + * @author Phillip Webb + * @author Yanming Zhou + * @author Phillip Webb + */ +record RedisUrl(URI uri, boolean useSsl, Credentials credentials, int database) { + + static RedisUrl of(String url) { + return (url != null) ? of(toUri(url)) : null; + } + + private static RedisUrl of(URI uri) { + boolean useSsl = ("rediss".equals(uri.getScheme())); + Credentials credentials = Credentials.fromUserInfo(uri.getUserInfo()); + int database = getDatabase(uri); + return new RedisUrl(uri, useSsl, credentials, database); + } + + private static int getDatabase(URI uri) { + String path = uri.getPath(); + String[] split = (!StringUtils.hasText(path)) ? new String[0] : path.split("/", 2); + return (split.length > 1 && !split[1].isEmpty()) ? Integer.parseInt(split[1]) : 0; + } + + private static URI toUri(String url) { + try { + URI uri = new URI(url); + String scheme = uri.getScheme(); + if (!"redis".equals(scheme) && !"rediss".equals(scheme)) { + throw new RedisUrlSyntaxException(url); + } + return uri; + } + catch (URISyntaxException ex) { + throw new RedisUrlSyntaxException(url, ex); + } + } + + /** + * Redis connection credentials. + * + * @param username the username or {@code null} + * @param password the password + */ + record Credentials(String username, String password) { + + private static final Credentials NONE = new Credentials(null, null); + + private static Credentials fromUserInfo(String userInfo) { + if (userInfo == null) { + return NONE; + } + int index = userInfo.indexOf(':'); + if (index != -1) { + return new Credentials(userInfo.substring(0, index), userInfo.substring(index + 1)); + } + return new Credentials(null, userInfo); + } + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/rest/RepositoryRestProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/rest/RepositoryRestProperties.java index ce4397ab10be..be501156e66b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/rest/RepositoryRestProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/data/rest/RepositoryRestProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,7 +28,7 @@ * @author Stephane Nicoll * @since 1.3.0 */ -@ConfigurationProperties(prefix = "spring.data.rest") +@ConfigurationProperties("spring.data.rest") public class RepositoryRestProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/diagnostics/analyzer/NoSuchBeanDefinitionFailureAnalyzer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/diagnostics/analyzer/NoSuchBeanDefinitionFailureAnalyzer.java index 10d5f2159614..96a4d0f34b1b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/diagnostics/analyzer/NoSuchBeanDefinitionFailureAnalyzer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/diagnostics/analyzer/NoSuchBeanDefinitionFailureAnalyzer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -65,7 +65,8 @@ class NoSuchBeanDefinitionFailureAnalyzer extends AbstractInjectionFailureAnalyz private final ConditionEvaluationReport report; NoSuchBeanDefinitionFailureAnalyzer(BeanFactory beanFactory) { - Assert.isInstanceOf(ConfigurableListableBeanFactory.class, beanFactory); + Assert.isTrue(beanFactory instanceof ConfigurableListableBeanFactory, + "'beanFactory' must be a ConfigurableListableBeanFactory"); this.beanFactory = (ConfigurableListableBeanFactory) beanFactory; this.metadataReaderFactory = new CachingMetadataReaderFactory(this.beanFactory.getBeanClassLoader()); // Get early as won't be accessible once context has failed to start diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/domain/EntityScanPackages.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/domain/EntityScanPackages.java index 82c0ce9178f3..0da1e3229844 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/domain/EntityScanPackages.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/domain/EntityScanPackages.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -96,8 +96,8 @@ public static EntityScanPackages get(BeanFactory beanFactory) { * @param packageNames the package names to register */ public static void register(BeanDefinitionRegistry registry, String... packageNames) { - Assert.notNull(registry, "Registry must not be null"); - Assert.notNull(packageNames, "PackageNames must not be null"); + Assert.notNull(registry, "'registry' must not be null"); + Assert.notNull(packageNames, "'packageNames' must not be null"); register(registry, Arrays.asList(packageNames)); } @@ -107,8 +107,8 @@ public static void register(BeanDefinitionRegistry registry, String... packageNa * @param packageNames the package names to register */ public static void register(BeanDefinitionRegistry registry, Collection packageNames) { - Assert.notNull(registry, "Registry must not be null"); - Assert.notNull(packageNames, "PackageNames must not be null"); + Assert.notNull(registry, "'registry' must not be null"); + Assert.notNull(packageNames, "'packageNames' must not be null"); if (registry.containsBeanDefinition(BEAN)) { EntityScanPackagesBeanDefinition beanDefinition = (EntityScanPackagesBeanDefinition) registry .getBeanDefinition(BEAN); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/domain/EntityScanner.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/domain/EntityScanner.java index 4fb4efd0a9d3..8b291eb82fa8 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/domain/EntityScanner.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/domain/EntityScanner.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ public class EntityScanner { * @param context the source application context */ public EntityScanner(ApplicationContext context) { - Assert.notNull(context, "Context must not be null"); + Assert.notNull(context, "'context' must not be null"); this.context = context; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/elasticsearch/ElasticsearchConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/elasticsearch/ElasticsearchConnectionDetails.java index 33c499ebaad7..eef66ee7cc5f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/elasticsearch/ElasticsearchConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/elasticsearch/ElasticsearchConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,6 +21,7 @@ import java.util.List; import org.springframework.boot.autoconfigure.service.connection.ConnectionDetails; +import org.springframework.boot.ssl.SslBundle; /** * Details required to establish a connection to an Elasticsearch service. @@ -63,6 +64,15 @@ default String getPathPrefix() { return null; } + /** + * SSL bundle to use. + * @return the SSL bundle to use + * @since 3.5.0 + */ + default SslBundle getSslBundle() { + return null; + } + /** * An Elasticsearch node. * diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/elasticsearch/ElasticsearchRestClientConfigurations.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/elasticsearch/ElasticsearchRestClientConfigurations.java index b18a14cb4d95..79cd46cdb0e8 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/elasticsearch/ElasticsearchRestClientConfigurations.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/elasticsearch/ElasticsearchRestClientConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -44,12 +44,14 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnSingleCandidate; import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchConnectionDetails.Node; import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchConnectionDetails.Node.Protocol; +import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchProperties.Restclient.Ssl; import org.springframework.boot.context.properties.PropertyMapper; import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.ssl.SslBundles; import org.springframework.boot.ssl.SslOptions; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.util.Assert; import org.springframework.util.StringUtils; /** @@ -75,8 +77,8 @@ static class RestClientBuilderConfiguration { @Bean @ConditionalOnMissingBean(ElasticsearchConnectionDetails.class) - PropertiesElasticsearchConnectionDetails elasticsearchConnectionDetails() { - return new PropertiesElasticsearchConnectionDetails(this.properties); + PropertiesElasticsearchConnectionDetails elasticsearchConnectionDetails(ObjectProvider sslBundles) { + return new PropertiesElasticsearchConnectionDetails(this.properties, sslBundles.getIfAvailable()); } @Bean @@ -87,16 +89,16 @@ RestClientBuilderCustomizer defaultRestClientBuilderCustomizer( @Bean RestClientBuilder elasticsearchRestClientBuilder(ElasticsearchConnectionDetails connectionDetails, - ObjectProvider builderCustomizers, ObjectProvider sslBundles) { + ObjectProvider builderCustomizers) { RestClientBuilder builder = RestClient.builder(connectionDetails.getNodes() .stream() .map((node) -> new HttpHost(node.hostname(), node.port(), node.protocol().getScheme())) .toArray(HttpHost[]::new)); builder.setHttpClientConfigCallback((httpClientBuilder) -> { builderCustomizers.orderedStream().forEach((customizer) -> customizer.customize(httpClientBuilder)); - String sslBundleName = this.properties.getRestclient().getSsl().getBundle(); - if (StringUtils.hasText(sslBundleName)) { - configureSsl(httpClientBuilder, sslBundles.getObject().getBundle(sslBundleName)); + SslBundle sslBundle = connectionDetails.getSslBundle(); + if (sslBundle != null) { + configureSsl(httpClientBuilder, sslBundle); } return httpClientBuilder; }); @@ -236,8 +238,11 @@ static class PropertiesElasticsearchConnectionDetails implements ElasticsearchCo private final ElasticsearchProperties properties; - PropertiesElasticsearchConnectionDetails(ElasticsearchProperties properties) { + private final SslBundles sslBundles; + + PropertiesElasticsearchConnectionDetails(ElasticsearchProperties properties, SslBundles sslBundles) { this.properties = properties; + this.sslBundles = sslBundles; } @Override @@ -260,6 +265,16 @@ public String getPathPrefix() { return this.properties.getPathPrefix(); } + @Override + public SslBundle getSslBundle() { + Ssl ssl = this.properties.getRestclient().getSsl(); + if (StringUtils.hasLength(ssl.getBundle())) { + Assert.notNull(this.sslBundles, "SSL bundle name has been set but no SSL bundles found in context"); + return this.sslBundles.getBundle(ssl.getBundle()); + } + return null; + } + private Node createNode(String uri) { if (!(uri.startsWith("http://") || uri.startsWith("https://"))) { uri = "http://" + uri; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayAutoConfiguration.java index 65780ae78171..be73b2b392a0 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayAutoConfiguration.java @@ -46,6 +46,7 @@ import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.AnyNestedCondition; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; @@ -105,7 +106,7 @@ HibernateJpaAutoConfiguration.class }) @ConditionalOnClass(Flyway.class) @Conditional(FlywayDataSourceCondition.class) -@ConditionalOnProperty(prefix = "spring.flyway", name = "enabled", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.flyway.enabled", matchIfMissing = true) @Import(DatabaseInitializationDependencyConfigurer.class) @ImportRuntimeHints(FlywayAutoConfigurationRuntimeHints.class) public class FlywayAutoConfiguration { @@ -449,7 +450,7 @@ private static final class JdbcConnectionDetailsCondition { } - @ConditionalOnProperty(prefix = "spring.flyway", name = "url") + @ConditionalOnProperty("spring.flyway.url") private static final class FlywayUrlCondition { } @@ -581,7 +582,7 @@ static class Extension { Extension(FluentConfiguration configuration, Class type, String name) { this.extension = SingletonSupplier.of(() -> { E extension = configuration.getPluginRegister().getPlugin(type); - Assert.notNull(extension, () -> "Flyway %s extension missing".formatted(name)); + Assert.state(extension != null, () -> "Flyway %s extension missing".formatted(name)); return extension; }); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayMigrationInitializer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayMigrationInitializer.java index 444270cb05f5..cdd3f1f1d624 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayMigrationInitializer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayMigrationInitializer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -51,7 +51,7 @@ public FlywayMigrationInitializer(Flyway flyway) { * @param migrationStrategy the migration strategy or {@code null} */ public FlywayMigrationInitializer(Flyway flyway, FlywayMigrationStrategy migrationStrategy) { - Assert.notNull(flyway, "Flyway must not be null"); + Assert.notNull(flyway, "'flyway' must not be null"); this.flyway = flyway; this.migrationStrategy = migrationStrategy; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayProperties.java index fa828f5b8985..e5d8e9ac2abc 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/flyway/FlywayProperties.java @@ -40,7 +40,7 @@ * @author Chris Bono * @since 1.1.0 */ -@ConfigurationProperties(prefix = "spring.flyway") +@ConfigurationProperties("spring.flyway") public class FlywayProperties { /** @@ -596,7 +596,7 @@ public void setCleanDisabled(boolean cleanDisabled) { } @Deprecated(since = "3.4.0", forRemoval = true) - @DeprecatedConfigurationProperty(since = "3.4.0", reason = "Deprecated in Flyway 10.18") + @DeprecatedConfigurationProperty(since = "3.4.0", reason = "Deprecated in Flyway 10.18 and removed in Flyway 11.0") public boolean isCleanOnValidationError() { return this.cleanOnValidationError; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerProperties.java index e25c7da057e4..9b8615719b5c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Andy Wilkinson * @since 1.1.0 */ -@ConfigurationProperties(prefix = "spring.freemarker") +@ConfigurationProperties("spring.freemarker") public class FreeMarkerProperties extends AbstractTemplateViewResolverProperties { public static final String DEFAULT_TEMPLATE_LOADER_PATH = "classpath:/templates/"; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerReactiveWebConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerReactiveWebConfiguration.java index 383fc1fd65bc..009ede2484ca 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerReactiveWebConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerReactiveWebConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,8 +18,8 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfigureAfter; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.web.reactive.WebFluxAutoConfiguration; import org.springframework.context.annotation.Bean; @@ -59,7 +59,7 @@ freemarker.template.Configuration freeMarkerConfiguration(FreeMarkerConfig confi @Bean @ConditionalOnMissingBean(name = "freeMarkerViewResolver") - @ConditionalOnProperty(name = "spring.freemarker.enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.freemarker.enabled", matchIfMissing = true) FreeMarkerViewResolver freeMarkerViewResolver() { FreeMarkerViewResolver resolver = new FreeMarkerViewResolver(); resolver.setPrefix(getProperties().getPrefix()); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerServletWebConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerServletWebConfiguration.java index 6b9b8ae790ba..123bb1db6aec 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerServletWebConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/freemarker/FreeMarkerServletWebConfiguration.java @@ -21,9 +21,9 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfigureAfter; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.web.ConditionalOnEnabledResourceChain; import org.springframework.boot.autoconfigure.web.servlet.ConditionalOnMissingFilterBean; @@ -68,7 +68,7 @@ freemarker.template.Configuration freeMarkerConfiguration(FreeMarkerConfig confi @Bean @ConditionalOnMissingBean(name = "freeMarkerViewResolver") - @ConditionalOnProperty(name = "spring.freemarker.enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.freemarker.enabled", matchIfMissing = true) FreeMarkerViewResolver freeMarkerViewResolver() { FreeMarkerViewResolver resolver = new FreeMarkerViewResolver(); getProperties().applyToMvcViewResolver(resolver); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/DefaultGraphQlSchemaCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/DefaultGraphQlSchemaCondition.java index 77ead8392c01..d9a4aa84a259 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/DefaultGraphQlSchemaCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/DefaultGraphQlSchemaCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,13 +34,15 @@ import org.springframework.core.io.support.ResourcePatternResolver; import org.springframework.core.io.support.ResourcePatternUtils; import org.springframework.core.type.AnnotatedTypeMetadata; +import org.springframework.graphql.execution.GraphQlSource; /** * {@link Condition} that checks whether a GraphQL schema has been defined in the * application. This is looking for: *

* * @author Brian Clozel @@ -82,6 +84,14 @@ public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeM else { messages.add((message.didNotFind("GraphQlSourceBuilderCustomizer").atAll())); } + String[] graphQlSourceBeanNames = beanFactory.getBeanNamesForType(GraphQlSource.class, false, false); + if (graphQlSourceBeanNames.length != 0) { + match = true; + messages.add(message.found("GraphQlSource").items(Arrays.asList(graphQlSourceBeanNames))); + } + else { + messages.add((message.didNotFind("GraphQlSource").atAll())); + } return new ConditionOutcome(match, ConditionMessage.of(messages)); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlAutoConfiguration.java index 599b613ad4a5..2f997c8fb5f3 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -96,11 +96,15 @@ public GraphQlSource graphQlSource(ResourcePatternResolver resourcePatternResolv ObjectProvider subscriptionExceptionResolvers, ObjectProvider instrumentations, ObjectProvider wiringConfigurers, ObjectProvider sourceCustomizers) { + String[] schemaLocations = properties.getSchema().getLocations(); - Resource[] schemaResources = resolveSchemaResources(resourcePatternResolver, schemaLocations, - properties.getSchema().getFileExtensions()); + List schemaResources = new ArrayList<>(); + schemaResources.addAll(resolveSchemaResources(resourcePatternResolver, schemaLocations, + properties.getSchema().getFileExtensions())); + schemaResources.addAll(Arrays.asList(properties.getSchema().getAdditionalFiles())); + GraphQlSource.SchemaResourceBuilder builder = GraphQlSource.schemaResourceBuilder() - .schemaResources(schemaResources) + .schemaResources(schemaResources.toArray(new Resource[0])) .exceptionResolvers(exceptionResolvers.orderedStream().toList()) .subscriptionExceptionResolvers(subscriptionExceptionResolvers.orderedStream().toList()) .instrumentation(instrumentations.orderedStream().toList()); @@ -116,7 +120,7 @@ public GraphQlSource graphQlSource(ResourcePatternResolver resourcePatternResolv return builder.build(); } - private Resource[] resolveSchemaResources(ResourcePatternResolver resolver, String[] locations, + private List resolveSchemaResources(ResourcePatternResolver resolver, String[] locations, String[] extensions) { List resources = new ArrayList<>(); for (String location : locations) { @@ -124,7 +128,7 @@ private Resource[] resolveSchemaResources(ResourcePatternResolver resolver, Stri resources.addAll(resolveSchemaResources(resolver, location + "*" + extension)); } } - return resources.toArray(new Resource[0]); + return resources; } private List resolveSchemaResources(ResourcePatternResolver resolver, String pattern) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlCorsProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlCorsProperties.java index ee9dbc1896a0..3d50e34399b2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlCorsProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlCorsProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * @author Brian Clozel * @since 2.7.0 */ -@ConfigurationProperties(prefix = "spring.graphql.cors") +@ConfigurationProperties("spring.graphql.cors") public class GraphQlCorsProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlProperties.java index 4324d47668fd..be698e1c963b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/GraphQlProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,6 +20,8 @@ import java.util.Arrays; import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.context.properties.DeprecatedConfigurationProperty; +import org.springframework.core.io.Resource; /** * {@link ConfigurationProperties Properties} for Spring GraphQL. @@ -27,34 +29,38 @@ * @author Brian Clozel * @since 2.7.0 */ -@ConfigurationProperties(prefix = "spring.graphql") +@ConfigurationProperties("spring.graphql") public class GraphQlProperties { - /** - * Path at which to expose a GraphQL request HTTP endpoint. - */ - private String path = "/graphql"; + private final Http http = new Http(); private final Graphiql graphiql = new Graphiql(); + private final Rsocket rsocket = new Rsocket(); + private final Schema schema = new Schema(); - private final Websocket websocket = new Websocket(); + private final DeprecatedSse sse = new DeprecatedSse(this.http.getSse()); - private final Rsocket rsocket = new Rsocket(); + private final Websocket websocket = new Websocket(); - private final Sse sse = new Sse(); + public Http getHttp() { + return this.http; + } public Graphiql getGraphiql() { return this.graphiql; } + @DeprecatedConfigurationProperty(replacement = "spring.graphql.http.path", since = "3.5.0") + @Deprecated(since = "3.5.0", forRemoval = true) public String getPath() { - return this.path; + return getHttp().getPath(); } + @Deprecated(since = "3.5.0", forRemoval = true) public void setPath(String path) { - this.path = path; + getHttp().setPath(path); } public Schema getSchema() { @@ -69,10 +75,33 @@ public Rsocket getRsocket() { return this.rsocket; } - public Sse getSse() { + public DeprecatedSse getSse() { return this.sse; } + public static class Http { + + /** + * Path at which to expose a GraphQL request HTTP endpoint. + */ + private String path = "/graphql"; + + private final Sse sse = new Sse(); + + public String getPath() { + return this.path; + } + + public void setPath(String path) { + this.path = path; + } + + public Sse getSse() { + return this.sse; + } + + } + public static class Schema { /** @@ -85,6 +114,11 @@ public static class Schema { */ private String[] fileExtensions = new String[] { ".graphqls", ".gqls" }; + /** + * Locations of additional, individual schema files to parse. + */ + private Resource[] additionalFiles = new Resource[0]; + private final Inspection inspection = new Inspection(); private final Introspection introspection = new Introspection(); @@ -107,6 +141,14 @@ public void setFileExtensions(String[] fileExtensions) { this.fileExtensions = fileExtensions; } + public Resource[] getAdditionalFiles() { + return this.additionalFiles; + } + + public void setAdditionalFiles(Resource[] additionalFiles) { + this.additionalFiles = additionalFiles; + } + private String[] appendSlashIfNecessary(String[] locations) { return Arrays.stream(locations) .map((location) -> location.endsWith("/") ? location : location + "/") @@ -164,7 +206,7 @@ public static class Printer { /** * Whether the endpoint that prints the schema is enabled. Schema is available - * under spring.graphql.path + "/schema". + * under spring.graphql.http.path + "/schema". */ private boolean enabled = false; @@ -273,11 +315,24 @@ public void setMapping(String mapping) { public static class Sse { + /** + * How frequently keep-alive messages should be sent. + */ + private Duration keepAlive; + /** * Time required for concurrent handling to complete. */ private Duration timeout; + public Duration getKeepAlive() { + return this.keepAlive; + } + + public void setKeepAlive(Duration keepAlive) { + this.keepAlive = keepAlive; + } + public Duration getTimeout() { return this.timeout; } @@ -288,4 +343,25 @@ public void setTimeout(Duration timeout) { } + public static class DeprecatedSse { + + private final Sse sse; + + public DeprecatedSse(Sse sse) { + this.sse = sse; + } + + @DeprecatedConfigurationProperty(replacement = "spring.graphql.http.sse.timeout", since = "3.5.0") + @Deprecated(since = "3.5.0", forRemoval = true) + public Duration getTimeout() { + return this.sse.getTimeout(); + } + + @Deprecated(since = "3.5.0", forRemoval = true) + public void setTimeout(Duration timeout) { + this.sse.setTimeout(timeout); + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/reactive/GraphQlWebFluxAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/reactive/GraphQlWebFluxAutoConfiguration.java index 05f6eaadcf1c..8dddf6d5ae4a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/reactive/GraphQlWebFluxAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/reactive/GraphQlWebFluxAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -87,28 +87,30 @@ public class GraphQlWebFluxAutoConfiguration { @Bean @ConditionalOnMissingBean - public WebGraphQlHandler webGraphQlHandler(ExecutionGraphQlService service, - ObjectProvider interceptors) { - return WebGraphQlHandler.builder(service).interceptors(interceptors.orderedStream().toList()).build(); + public GraphQlHttpHandler graphQlHttpHandler(WebGraphQlHandler webGraphQlHandler) { + return new GraphQlHttpHandler(webGraphQlHandler); } @Bean @ConditionalOnMissingBean - public GraphQlHttpHandler graphQlHttpHandler(WebGraphQlHandler webGraphQlHandler) { - return new GraphQlHttpHandler(webGraphQlHandler); + public GraphQlSseHandler graphQlSseHandler(WebGraphQlHandler webGraphQlHandler, GraphQlProperties properties) { + return new GraphQlSseHandler(webGraphQlHandler, properties.getHttp().getSse().getTimeout(), + properties.getHttp().getSse().getKeepAlive()); } @Bean @ConditionalOnMissingBean - public GraphQlSseHandler graphQlSseHandler(WebGraphQlHandler webGraphQlHandler) { - return new GraphQlSseHandler(webGraphQlHandler); + public WebGraphQlHandler webGraphQlHandler(ExecutionGraphQlService service, + ObjectProvider interceptors) { + return WebGraphQlHandler.builder(service).interceptors(interceptors.orderedStream().toList()).build(); } @Bean @Order(0) public RouterFunction graphQlRouterFunction(GraphQlHttpHandler httpHandler, - GraphQlSseHandler sseHandler, GraphQlSource graphQlSource, GraphQlProperties properties) { - String path = properties.getPath(); + GraphQlSseHandler sseHandler, ObjectProvider graphQlSourceProvider, + GraphQlProperties properties) { + String path = properties.getHttp().getPath(); logger.info(LogMessage.format("GraphQL endpoint HTTP POST %s", path)); RouterFunctions.Builder builder = RouterFunctions.route(); builder.route(GraphQlRequestPredicates.graphQlHttp(path), httpHandler::handleRequest); @@ -119,7 +121,8 @@ public RouterFunction graphQlRouterFunction(GraphQlHttpHandler h GraphiQlHandler graphQlHandler = new GraphiQlHandler(path, properties.getWebsocket().getPath()); builder.GET(properties.getGraphiql().getPath(), graphQlHandler::handleRequest); } - if (properties.getSchema().getPrinter().isEnabled()) { + GraphQlSource graphQlSource = graphQlSourceProvider.getIfAvailable(); + if (properties.getSchema().getPrinter().isEnabled() && graphQlSource != null) { SchemaHandler schemaHandler = new SchemaHandler(graphQlSource); builder.GET(path + "/schema", schemaHandler::handleRequest); } @@ -158,14 +161,14 @@ public GraphQlEndpointCorsConfiguration(GraphQlProperties graphQlProps, GraphQlC public void addCorsMappings(CorsRegistry registry) { CorsConfiguration configuration = this.corsProperties.toCorsConfiguration(); if (configuration != null) { - registry.addMapping(this.graphQlProperties.getPath()).combine(configuration); + registry.addMapping(this.graphQlProperties.getHttp().getPath()).combine(configuration); } } } @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.graphql.websocket", name = "path") + @ConditionalOnProperty("spring.graphql.websocket.path") public static class WebSocketConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/rsocket/GraphQlRSocketAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/rsocket/GraphQlRSocketAutoConfiguration.java index 6114714d3c21..b6345a448432 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/rsocket/GraphQlRSocketAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/rsocket/GraphQlRSocketAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,7 @@ @AutoConfiguration(after = { GraphQlAutoConfiguration.class, RSocketMessagingAutoConfiguration.class }) @ConditionalOnClass({ GraphQL.class, GraphQlSource.class, RSocketServer.class, HttpServer.class }) @ConditionalOnBean({ RSocketMessageHandler.class, AnnotatedControllerConfigurer.class }) -@ConditionalOnProperty(prefix = "spring.graphql.rsocket", name = "mapping") +@ConditionalOnProperty("spring.graphql.rsocket.mapping") public class GraphQlRSocketAutoConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/servlet/GraphQlWebMvcAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/servlet/GraphQlWebMvcAutoConfiguration.java index f1004026d806..110e2a5afb1c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/servlet/GraphQlWebMvcAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/graphql/servlet/GraphQlWebMvcAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,6 @@ import org.springframework.boot.autoconfigure.graphql.GraphQlAutoConfiguration; import org.springframework.boot.autoconfigure.graphql.GraphQlCorsProperties; import org.springframework.boot.autoconfigure.graphql.GraphQlProperties; -import org.springframework.boot.autoconfigure.graphql.GraphQlProperties.Sse; import org.springframework.boot.autoconfigure.http.HttpMessageConverters; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; @@ -99,8 +98,8 @@ public GraphQlHttpHandler graphQlHttpHandler(WebGraphQlHandler webGraphQlHandler @Bean @ConditionalOnMissingBean public GraphQlSseHandler graphQlSseHandler(WebGraphQlHandler webGraphQlHandler, GraphQlProperties properties) { - Sse sse = properties.getSse(); - return new GraphQlSseHandler(webGraphQlHandler, sse.getTimeout()); + return new GraphQlSseHandler(webGraphQlHandler, properties.getHttp().getSse().getTimeout(), + properties.getHttp().getSse().getKeepAlive()); } @Bean @@ -113,8 +112,9 @@ public WebGraphQlHandler webGraphQlHandler(ExecutionGraphQlService service, @Bean @Order(0) public RouterFunction graphQlRouterFunction(GraphQlHttpHandler httpHandler, - GraphQlSseHandler sseHandler, GraphQlSource graphQlSource, GraphQlProperties properties) { - String path = properties.getPath(); + GraphQlSseHandler sseHandler, ObjectProvider graphQlSourceProvider, + GraphQlProperties properties) { + String path = properties.getHttp().getPath(); logger.info(LogMessage.format("GraphQL endpoint HTTP POST %s", path)); RouterFunctions.Builder builder = RouterFunctions.route(); builder.route(GraphQlRequestPredicates.graphQlHttp(path), httpHandler::handleRequest); @@ -125,7 +125,8 @@ public RouterFunction graphQlRouterFunction(GraphQlHttpHandler h GraphiQlHandler graphiQLHandler = new GraphiQlHandler(path, properties.getWebsocket().getPath()); builder.GET(properties.getGraphiql().getPath(), graphiQLHandler::handleRequest); } - if (properties.getSchema().getPrinter().isEnabled()) { + GraphQlSource graphQlSource = graphQlSourceProvider.getIfAvailable(); + if (properties.getSchema().getPrinter().isEnabled() && graphQlSource != null) { SchemaHandler schemaHandler = new SchemaHandler(graphQlSource); builder.GET(path + "/schema", schemaHandler::handleRequest); } @@ -164,7 +165,7 @@ public GraphQlEndpointCorsConfiguration(GraphQlProperties graphQlProps, GraphQlC public void addCorsMappings(CorsRegistry registry) { CorsConfiguration configuration = this.corsProperties.toCorsConfiguration(); if (configuration != null) { - registry.addMapping(this.graphQlProperties.getPath()).combine(configuration); + registry.addMapping(this.graphQlProperties.getHttp().getPath()).combine(configuration); } } @@ -172,7 +173,7 @@ public void addCorsMappings(CorsRegistry registry) { @Configuration(proxyBeanMethods = false) @ConditionalOnClass({ ServerContainer.class, WebSocketHandler.class }) - @ConditionalOnProperty(prefix = "spring.graphql.websocket", name = "path") + @ConditionalOnProperty("spring.graphql.websocket.path") public static class WebSocketConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/groovy/template/GroovyTemplateAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/groovy/template/GroovyTemplateAutoConfiguration.java index fe1b9dea2806..cd367c140ce2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/groovy/template/GroovyTemplateAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/groovy/template/GroovyTemplateAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,19 +26,22 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.boot.autoconfigure.template.TemplateLocation; import org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfiguration; -import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.boot.context.properties.PropertyMapper; +import org.springframework.boot.context.properties.bind.Bindable; +import org.springframework.boot.context.properties.bind.Binder; import org.springframework.context.ApplicationContext; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.i18n.LocaleContextHolder; +import org.springframework.core.env.Environment; import org.springframework.core.log.LogMessage; import org.springframework.web.servlet.view.UrlBasedViewResolver; import org.springframework.web.servlet.view.groovy.GroovyMarkupConfig; @@ -109,11 +112,23 @@ private boolean isUsingGroovyAllJar() { @Bean @ConditionalOnMissingBean(GroovyMarkupConfig.class) - @ConfigurationProperties(prefix = "spring.groovy.template.configuration") - public GroovyMarkupConfigurer groovyMarkupConfigurer(ObjectProvider templateEngine) { + GroovyMarkupConfigurer groovyMarkupConfigurer(ObjectProvider templateEngine, + Environment environment) { GroovyMarkupConfigurer configurer = new GroovyMarkupConfigurer(); - configurer.setResourceLoaderPath(this.properties.getResourceLoaderPath()); - configurer.setCacheTemplates(this.properties.isCache()); + PropertyMapper map = PropertyMapper.get(); + map.from(this.properties::isAutoEscape).to(configurer::setAutoEscape); + map.from(this.properties::isAutoIndent).to(configurer::setAutoIndent); + map.from(this.properties::getAutoIndentString).to(configurer::setAutoIndentString); + map.from(this.properties::isAutoNewLine).to(configurer::setAutoNewLine); + map.from(this.properties::getBaseTemplateClass).to(configurer::setBaseTemplateClass); + map.from(this.properties::isCache).to(configurer::setCacheTemplates); + map.from(this.properties::getDeclarationEncoding).to(configurer::setDeclarationEncoding); + map.from(this.properties::isExpandEmptyElements).to(configurer::setExpandEmptyElements); + map.from(this.properties::getLocale).to(configurer::setLocale); + map.from(this.properties::getNewLineString).to(configurer::setNewLineString); + map.from(this.properties::getResourceLoaderPath).to(configurer::setResourceLoaderPath); + map.from(this.properties::isUseDoubleQuotes).to(configurer::setUseDoubleQuotes); + Binder.get(environment).bind("spring.groovy.template.configuration", Bindable.ofInstance(configurer)); templateEngine.ifAvailable(configurer::setTemplateEngine); return configurer; } @@ -123,7 +138,7 @@ public GroovyMarkupConfigurer groovyMarkupConfigurer(ObjectProvider baseTemplateClass = BaseTemplate.class; + + /** + * Encoding used to write the declaration heading. + */ + private String declarationEncoding; + + /** + * Whether elements without a body should be written expanded (<br></br>) + * or not (<br/>). + */ + private boolean expandEmptyElements; + + /** + * Default locale for template resolution. + */ + private Locale locale; + + /** + * String used to write a new line. Defaults to the system's line separator. + */ + private String newLineString; + /** * Template path. */ private String resourceLoaderPath = DEFAULT_RESOURCE_LOADER_PATH; + /** + * Whether attributes should use double quotes. + */ + private boolean useDoubleQuotes; + public GroovyTemplateProperties() { super(DEFAULT_PREFIX, DEFAULT_SUFFIX); setRequestContextAttribute(DEFAULT_REQUEST_CONTEXT_ATTRIBUTE); } + public boolean isAutoEscape() { + return this.autoEscape; + } + + public void setAutoEscape(boolean autoEscape) { + this.autoEscape = autoEscape; + } + + public boolean isAutoIndent() { + return this.autoIndent; + } + + public void setAutoIndent(boolean autoIndent) { + this.autoIndent = autoIndent; + } + + public String getAutoIndentString() { + return this.autoIndentString; + } + + public void setAutoIndentString(String autoIndentString) { + this.autoIndentString = autoIndentString; + } + + public boolean isAutoNewLine() { + return this.autoNewLine; + } + + public void setAutoNewLine(boolean autoNewLine) { + this.autoNewLine = autoNewLine; + } + + public Class getBaseTemplateClass() { + return this.baseTemplateClass; + } + + public void setBaseTemplateClass(Class baseTemplateClass) { + this.baseTemplateClass = baseTemplateClass; + } + + public String getDeclarationEncoding() { + return this.declarationEncoding; + } + + public void setDeclarationEncoding(String declarationEncoding) { + this.declarationEncoding = declarationEncoding; + } + + public boolean isExpandEmptyElements() { + return this.expandEmptyElements; + } + + public void setExpandEmptyElements(boolean expandEmptyElements) { + this.expandEmptyElements = expandEmptyElements; + } + + public Locale getLocale() { + return this.locale; + } + + public void setLocale(Locale locale) { + this.locale = locale; + } + + public String getNewLineString() { + return this.newLineString; + } + + public void setNewLineString(String newLineString) { + this.newLineString = newLineString; + } + public String getResourceLoaderPath() { return this.resourceLoaderPath; } @@ -56,4 +183,12 @@ public void setResourceLoaderPath(String resourceLoaderPath) { this.resourceLoaderPath = resourceLoaderPath; } + public boolean isUseDoubleQuotes() { + return this.useDoubleQuotes; + } + + public void setUseDoubleQuotes(boolean useDoubleQuotes) { + this.useDoubleQuotes = useDoubleQuotes; + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/gson/GsonProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/gson/GsonProperties.java index 1d0e50ca006f..a75cd862d135 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/gson/GsonProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/gson/GsonProperties.java @@ -29,7 +29,7 @@ * @author Ivan Golovko * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.gson") +@ConfigurationProperties("spring.gson") public class GsonProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/h2/H2ConsoleAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/h2/H2ConsoleAutoConfiguration.java index d76f10724790..730640699d4b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/h2/H2ConsoleAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/h2/H2ConsoleAutoConfiguration.java @@ -29,8 +29,8 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.boot.autoconfigure.h2.H2ConsoleProperties.Settings; @@ -52,7 +52,7 @@ @AutoConfiguration(after = DataSourceAutoConfiguration.class) @ConditionalOnWebApplication(type = Type.SERVLET) @ConditionalOnClass(JakartaWebServlet.class) -@ConditionalOnProperty(prefix = "spring.h2.console", name = "enabled", havingValue = "true") +@ConditionalOnBooleanProperty("spring.h2.console.enabled") @EnableConfigurationProperties(H2ConsoleProperties.class) public class H2ConsoleAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/h2/H2ConsoleProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/h2/H2ConsoleProperties.java index a1dc97a034fb..1f420eef95e1 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/h2/H2ConsoleProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/h2/H2ConsoleProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ * @author Stephane Nicoll * @since 1.3.0 */ -@ConfigurationProperties(prefix = "spring.h2.console") +@ConfigurationProperties("spring.h2.console") public class H2ConsoleProperties { /** @@ -47,13 +47,13 @@ public String getPath() { } public void setPath(String path) { - Assert.notNull(path, "Path must not be null"); - Assert.isTrue(path.length() > 1, "Path must have length greater than 1"); - Assert.isTrue(path.startsWith("/"), "Path must start with '/'"); + Assert.notNull(path, "'path' must not be null"); + Assert.isTrue(path.length() > 1, "'path' must have length greater than 1"); + Assert.isTrue(path.startsWith("/"), "'path' must start with '/'"); this.path = path; } - public boolean getEnabled() { + public boolean isEnabled() { return this.enabled; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hateoas/HateoasProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hateoas/HateoasProperties.java index a1a17215501e..d131564ed6d5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hateoas/HateoasProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hateoas/HateoasProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ * @author Andy Wilkinson * @since 1.2.1 */ -@ConfigurationProperties(prefix = "spring.hateoas") +@ConfigurationProperties("spring.hateoas") public class HateoasProperties { /** @@ -34,7 +34,7 @@ public class HateoasProperties { */ private boolean useHalAsDefaultJsonMediaType = true; - public boolean getUseHalAsDefaultJsonMediaType() { + public boolean isUseHalAsDefaultJsonMediaType() { return this.useHalAsDefaultJsonMediaType; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hateoas/HypermediaAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hateoas/HypermediaAutoConfiguration.java index 50c0507a85c6..f1bd7ba473b2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hateoas/HypermediaAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hateoas/HypermediaAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,9 +20,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.data.rest.RepositoryRestMvcAutoConfiguration; import org.springframework.boot.autoconfigure.http.HttpMessageConvertersAutoConfiguration; @@ -60,8 +60,7 @@ public class HypermediaAutoConfiguration { @Bean @ConditionalOnMissingBean @ConditionalOnClass(name = "com.fasterxml.jackson.databind.ObjectMapper") - @ConditionalOnProperty(prefix = "spring.hateoas", name = "use-hal-as-default-json-media-type", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.hateoas.use-hal-as-default-json-media-type", matchIfMissing = true) HalConfiguration applicationJsonHalConfiguration() { return new HalConfiguration().withMediaType(MediaType.APPLICATION_JSON); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastConfigResourceCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastConfigResourceCondition.java index af5054c9eb28..9ff61bf97dff 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastConfigResourceCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastConfigResourceCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,7 @@ public abstract class HazelcastConfigResourceCondition extends ResourceCondition protected HazelcastConfigResourceCondition(String configSystemProperty, String... resourceLocations) { super("Hazelcast", HAZELCAST_CONFIG_PROPERTY, resourceLocations); - Assert.notNull(configSystemProperty, "ConfigSystemProperty must not be null"); + Assert.notNull(configSystemProperty, "'configSystemProperty' must not be null"); this.configSystemProperty = configSystemProperty; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastProperties.java index 524dbbaf9c1a..98687ca86a84 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Stephane Nicoll * @since 1.3.0 */ -@ConfigurationProperties(prefix = "spring.hazelcast") +@ConfigurationProperties("spring.hazelcast") public class HazelcastProperties { /** @@ -52,7 +52,7 @@ public Resource resolveConfigLocation() { if (this.config == null) { return null; } - Assert.isTrue(this.config.exists(), + Assert.state(this.config.exists(), () -> "Hazelcast configuration does not exist '" + this.config.getDescription() + "'"); return this.config; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/ConditionalOnPreferredJsonMapper.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/ConditionalOnPreferredJsonMapper.java new file mode 100644 index 000000000000..a1ab26d50101 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/ConditionalOnPreferredJsonMapper.java @@ -0,0 +1,54 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.context.annotation.Conditional; + +/** + * {@link Conditional @Conditional} that matches based on the preferred JSON mapper. A + * preference is expressed using the {@code spring.http.converters.preferred-json-mapper} + * configuration property, falling back to the + * {@code spring.mvc.converters.preferred-json-mapper} configuration property. When no + * preference is expressed Jackson is preferred by default. + * + * @author Andy Wilkinson + */ +@Conditional(OnPreferredJsonMapperCondition.class) +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.METHOD }) +@Documented +@interface ConditionalOnPreferredJsonMapper { + + JsonMapper value(); + + enum JsonMapper { + + GSON, + + JACKSON, + + JSONB, + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/GsonHttpMessageConvertersConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/GsonHttpMessageConvertersConfiguration.java index 1eacbebd98ff..9aa1311c124e 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/GsonHttpMessageConvertersConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/GsonHttpMessageConvertersConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,8 +22,8 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.NoneNestedConditions; +import org.springframework.boot.autoconfigure.http.ConditionalOnPreferredJsonMapper.JsonMapper; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Conditional; import org.springframework.context.annotation.Configuration; @@ -61,8 +61,7 @@ private static class PreferGsonOrJacksonAndJsonbUnavailableCondition extends Any super(ConfigurationPhase.REGISTER_BEAN); } - @ConditionalOnProperty(name = HttpMessageConvertersAutoConfiguration.PREFERRED_MAPPER_PROPERTY, - havingValue = "gson") + @ConditionalOnPreferredJsonMapper(JsonMapper.GSON) static class GsonPreferred { } @@ -85,8 +84,7 @@ static class JacksonAvailable { } - @ConditionalOnProperty(name = HttpMessageConvertersAutoConfiguration.PREFERRED_MAPPER_PROPERTY, - havingValue = "jsonb") + @ConditionalOnPreferredJsonMapper(JsonMapper.JSONB) static class JsonbPreferred { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/HttpMessageConvertersAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/HttpMessageConvertersAutoConfiguration.java index fe50c8eab9c8..0aaa539eb8ba 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/HttpMessageConvertersAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/HttpMessageConvertersAutoConfiguration.java @@ -66,8 +66,6 @@ @ImportRuntimeHints(HttpMessageConvertersAutoConfigurationRuntimeHints.class) public class HttpMessageConvertersAutoConfiguration { - static final String PREFERRED_MAPPER_PROPERTY = "spring.mvc.converters.preferred-json-mapper"; - @Bean @ConditionalOnMissingBean public HttpMessageConverters messageConverters(ObjectProvider> converters) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/JacksonHttpMessageConvertersConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/JacksonHttpMessageConvertersConfiguration.java index 687cb29dfc55..b3bac31dcb10 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/JacksonHttpMessageConvertersConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/JacksonHttpMessageConvertersConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,7 +22,7 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.autoconfigure.http.ConditionalOnPreferredJsonMapper.JsonMapper; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder; @@ -40,8 +40,7 @@ class JacksonHttpMessageConvertersConfiguration { @Configuration(proxyBeanMethods = false) @ConditionalOnClass(ObjectMapper.class) @ConditionalOnBean(ObjectMapper.class) - @ConditionalOnProperty(name = HttpMessageConvertersAutoConfiguration.PREFERRED_MAPPER_PROPERTY, - havingValue = "jackson", matchIfMissing = true) + @ConditionalOnPreferredJsonMapper(JsonMapper.JACKSON) static class MappingJackson2HttpMessageConverterConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/JsonbHttpMessageConvertersConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/JsonbHttpMessageConvertersConfiguration.java index 14291e97eb2d..a4d71684e73b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/JsonbHttpMessageConvertersConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/JsonbHttpMessageConvertersConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,7 +22,7 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.autoconfigure.http.ConditionalOnPreferredJsonMapper.JsonMapper; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Conditional; import org.springframework.context.annotation.Configuration; @@ -60,8 +60,7 @@ private static class PreferJsonbOrMissingJacksonAndGsonCondition extends AnyNest super(ConfigurationPhase.REGISTER_BEAN); } - @ConditionalOnProperty(name = HttpMessageConvertersAutoConfiguration.PREFERRED_MAPPER_PROPERTY, - havingValue = "jsonb") + @ConditionalOnPreferredJsonMapper(JsonMapper.JSONB) static class JsonbPreferred { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/OnPreferredJsonMapperCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/OnPreferredJsonMapperCondition.java new file mode 100644 index 000000000000..3018e30def5c --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/OnPreferredJsonMapperCondition.java @@ -0,0 +1,76 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http; + +import java.util.Locale; + +import org.springframework.boot.autoconfigure.condition.ConditionMessage; +import org.springframework.boot.autoconfigure.condition.ConditionOutcome; +import org.springframework.boot.autoconfigure.condition.SpringBootCondition; +import org.springframework.boot.autoconfigure.http.ConditionalOnPreferredJsonMapper.JsonMapper; +import org.springframework.context.annotation.ConditionContext; +import org.springframework.core.env.Environment; +import org.springframework.core.type.AnnotatedTypeMetadata; + +/** + * {@link SpringBootCondition} for + * {@link ConditionalOnPreferredJsonMapper @ConditionalOnPreferredJsonMapper}. + * + * @author Andy Wilkinson + */ +class OnPreferredJsonMapperCondition extends SpringBootCondition { + + private static final String PREFERRED_MAPPER_PROPERTY = "spring.http.converters.preferred-json-mapper"; + + @Deprecated(since = "3.5.0", forRemoval = true) + private static final String DEPRECATED_PREFERRED_MAPPER_PROPERTY = "spring.mvc.converters.preferred-json-mapper"; + + @Override + public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeMetadata metadata) { + JsonMapper conditionMapper = metadata.getAnnotations() + .get(ConditionalOnPreferredJsonMapper.class) + .getEnum("value", JsonMapper.class); + ConditionOutcome outcome = getMatchOutcome(context.getEnvironment(), PREFERRED_MAPPER_PROPERTY, + conditionMapper); + if (outcome != null) { + return outcome; + } + outcome = getMatchOutcome(context.getEnvironment(), DEPRECATED_PREFERRED_MAPPER_PROPERTY, conditionMapper); + if (outcome != null) { + return outcome; + } + ConditionMessage message = ConditionMessage + .forCondition(ConditionalOnPreferredJsonMapper.class, conditionMapper.name()) + .because("no property was configured and Jackson is the default"); + return (conditionMapper == JsonMapper.JACKSON) ? ConditionOutcome.match(message) + : ConditionOutcome.noMatch(message); + } + + private ConditionOutcome getMatchOutcome(Environment environment, String key, JsonMapper conditionMapper) { + String property = environment.getProperty(key); + if (property == null) { + return null; + } + JsonMapper configuredMapper = JsonMapper.valueOf(property.toUpperCase(Locale.ROOT)); + ConditionMessage message = ConditionMessage + .forCondition(ConditionalOnPreferredJsonMapper.class, configuredMapper.name()) + .because("property '%s' had the value '%s'".formatted(key, property)); + return (configuredMapper == conditionMapper) ? ConditionOutcome.match(message) + : ConditionOutcome.noMatch(message); + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/AbstractHttpClientProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/AbstractHttpClientProperties.java new file mode 100644 index 000000000000..ea0c41791bad --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/AbstractHttpClientProperties.java @@ -0,0 +1,120 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.client; + +import java.time.Duration; + +import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.http.client.HttpClientSettings; +import org.springframework.boot.http.client.HttpRedirects; +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslBundles; +import org.springframework.util.StringUtils; + +/** + * Abstract base class for properties that directly or indirectly make use of a blocking + * or reactive HTTP client. + * + * @author Phillip Webb + * @since 3.5.0 + * @see HttpClientSettings + */ +public abstract class AbstractHttpClientProperties { + + /** + * Handling for HTTP redirects. + */ + private HttpRedirects redirects = HttpRedirects.FOLLOW_WHEN_POSSIBLE; + + /** + * Default connect timeout for a client HTTP request. + */ + private Duration connectTimeout; + + /** + * Default read timeout for a client HTTP request. + */ + private Duration readTimeout; + + /** + * Default SSL configuration for a client HTTP request. + */ + private final Ssl ssl = new Ssl(); + + public HttpRedirects getRedirects() { + return this.redirects; + } + + public void setRedirects(HttpRedirects redirects) { + this.redirects = redirects; + } + + public Duration getConnectTimeout() { + return this.connectTimeout; + } + + public void setConnectTimeout(Duration connectTimeout) { + this.connectTimeout = connectTimeout; + } + + public Duration getReadTimeout() { + return this.readTimeout; + } + + public void setReadTimeout(Duration readTimeout) { + this.readTimeout = readTimeout; + } + + public Ssl getSsl() { + return this.ssl; + } + + /** + * Return {@link HttpClientSettings} based on these properties. + * @param sslBundles a {@link SslBundles} provider + * @return the {@link HttpClientSettings} + */ + protected HttpClientSettings httpClientSettings(ObjectProvider sslBundles) { + return new HttpClientSettings(this.redirects, this.connectTimeout, this.readTimeout, sslBundle(sslBundles)); + } + + private SslBundle sslBundle(ObjectProvider sslBundles) { + String name = getSsl().getBundle(); + return (StringUtils.hasLength(name)) ? sslBundles.getObject().getBundle(name) : null; + } + + /** + * SSL configuration. + */ + public static class Ssl { + + /** + * SSL bundle to use. + */ + private String bundle; + + public String getBundle() { + return this.bundle; + } + + public void setBundle(String bundle) { + this.bundle = bundle; + } + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/AbstractHttpRequestFactoryProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/AbstractHttpRequestFactoryProperties.java new file mode 100644 index 000000000000..5c8ae4b43a8d --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/AbstractHttpRequestFactoryProperties.java @@ -0,0 +1,100 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.client; + +import java.util.function.Supplier; + +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.http.client.ClientHttpRequestFactoryBuilder; +import org.springframework.boot.http.client.ClientHttpRequestFactorySettings; +import org.springframework.http.client.ClientHttpRequestFactory; + +/** + * Base {@link ConfigurationProperties @ConfigurationProperties} for configuring a + * {@link ClientHttpRequestFactory}. + * + * @author Phillip Webb + * @since 3.5.0 + * @see ClientHttpRequestFactorySettings + */ +public abstract class AbstractHttpRequestFactoryProperties extends AbstractHttpClientProperties { + + /** + * Default factory used for a client HTTP request. + */ + private Factory factory; + + public Factory getFactory() { + return this.factory; + } + + public void setFactory(Factory factory) { + this.factory = factory; + } + + /** + * Return a {@link ClientHttpRequestFactoryBuilder} based on the properties. + * @return a {@link ClientHttpRequestFactoryBuilder} + */ + protected final ClientHttpRequestFactoryBuilder factoryBuilder() { + Factory factory = getFactory(); + return (factory != null) ? factory.builder() : ClientHttpRequestFactoryBuilder.detect(); + } + + /** + * Supported factory types. + */ + public enum Factory { + + /** + * Apache HttpComponents HttpClient. + */ + HTTP_COMPONENTS(ClientHttpRequestFactoryBuilder::httpComponents), + + /** + * Jetty's HttpClient. + */ + JETTY(ClientHttpRequestFactoryBuilder::jetty), + + /** + * Reactor-Netty. + */ + REACTOR(ClientHttpRequestFactoryBuilder::reactor), + + /** + * Java's HttpClient. + */ + JDK(ClientHttpRequestFactoryBuilder::jdk), + + /** + * Standard JDK facilities. + */ + SIMPLE(ClientHttpRequestFactoryBuilder::simple); + + private final Supplier> builderSupplier; + + Factory(Supplier> builderSupplier) { + this.builderSupplier = builderSupplier; + } + + ClientHttpRequestFactoryBuilder builder() { + return this.builderSupplier.get(); + } + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/ClientHttpRequestFactoryBuilderCustomizer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/ClientHttpRequestFactoryBuilderCustomizer.java new file mode 100644 index 000000000000..a7b731842c7d --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/ClientHttpRequestFactoryBuilderCustomizer.java @@ -0,0 +1,38 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.client; + +import org.springframework.boot.http.client.ClientHttpRequestFactoryBuilder; + +/** + * Customizer that can be used to modify the auto-configured + * {@link ClientHttpRequestFactoryBuilder} when its type matches. + * + * @param the builder type + * @author Phillip Webb + * @since 3.5.0 + */ +public interface ClientHttpRequestFactoryBuilderCustomizer> { + + /** + * Customize the given builder. + * @param builder the builder to customize + * @return the customized builder + */ + B customize(B builder); + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/HttpClientAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/HttpClientAutoConfiguration.java index faa6273442db..58df28341499 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/HttpClientAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/HttpClientAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,22 +16,25 @@ package org.springframework.boot.autoconfigure.http.client; +import java.util.List; + import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.http.client.HttpClientProperties.Factory; import org.springframework.boot.autoconfigure.ssl.SslAutoConfiguration; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.boot.http.client.ClientHttpRequestFactoryBuilder; import org.springframework.boot.http.client.ClientHttpRequestFactorySettings; -import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.http.client.ClientHttpRequestFactorySettings.Redirects; +import org.springframework.boot.http.client.HttpClientSettings; +import org.springframework.boot.http.client.HttpRedirects; import org.springframework.boot.ssl.SslBundles; +import org.springframework.boot.util.LambdaSafe; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Conditional; import org.springframework.http.client.ClientHttpRequestFactory; -import org.springframework.util.StringUtils; /** * {@link EnableAutoConfiguration Auto-configuration} for @@ -48,23 +51,36 @@ public class HttpClientAutoConfiguration { @Bean @ConditionalOnMissingBean - ClientHttpRequestFactoryBuilder clientHttpRequestFactoryBuilder(HttpClientProperties httpClientProperties) { - Factory factory = httpClientProperties.getFactory(); - return (factory != null) ? factory.builder() : ClientHttpRequestFactoryBuilder.detect(); + ClientHttpRequestFactoryBuilder clientHttpRequestFactoryBuilder(HttpClientProperties httpClientProperties, + ObjectProvider> clientHttpRequestFactoryBuilderCustomizers) { + ClientHttpRequestFactoryBuilder builder = httpClientProperties.factoryBuilder(); + return customize(builder, clientHttpRequestFactoryBuilderCustomizers.orderedStream().toList()); + } + + @SuppressWarnings("unchecked") + private ClientHttpRequestFactoryBuilder customize(ClientHttpRequestFactoryBuilder builder, + List> customizers) { + ClientHttpRequestFactoryBuilder[] builderReference = { builder }; + LambdaSafe.callbacks(ClientHttpRequestFactoryBuilderCustomizer.class, customizers, builderReference[0]) + .invoke((customizer) -> builderReference[0] = customizer.customize(builderReference[0])); + return builderReference[0]; } @Bean @ConditionalOnMissingBean ClientHttpRequestFactorySettings clientHttpRequestFactorySettings(HttpClientProperties httpClientProperties, ObjectProvider sslBundles) { - SslBundle sslBundle = getSslBundle(httpClientProperties.getSsl(), sslBundles); - return new ClientHttpRequestFactorySettings(httpClientProperties.getRedirects(), - httpClientProperties.getConnectTimeout(), httpClientProperties.getReadTimeout(), sslBundle); + HttpClientSettings settings = httpClientProperties.httpClientSettings(sslBundles); + return new ClientHttpRequestFactorySettings(asRequestFactoryRedirects(settings.redirects()), + settings.connectTimeout(), settings.readTimeout(), settings.sslBundle()); } - private SslBundle getSslBundle(HttpClientProperties.Ssl properties, ObjectProvider sslBundles) { - String name = properties.getBundle(); - return (StringUtils.hasLength(name)) ? sslBundles.getObject().getBundle(name) : null; + private Redirects asRequestFactoryRedirects(HttpRedirects redirects) { + return switch (redirects) { + case FOLLOW_WHEN_POSSIBLE -> Redirects.FOLLOW_WHEN_POSSIBLE; + case FOLLOW -> Redirects.FOLLOW; + case DONT_FOLLOW -> Redirects.DONT_FOLLOW; + }; } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/HttpClientProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/HttpClientProperties.java index abdc2e0c4c13..7542c9ba93b8 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/HttpClientProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/HttpClientProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,12 +16,8 @@ package org.springframework.boot.autoconfigure.http.client; -import java.time.Duration; -import java.util.function.Supplier; - import org.springframework.boot.context.properties.ConfigurationProperties; -import org.springframework.boot.http.client.ClientHttpRequestFactoryBuilder; -import org.springframework.boot.http.client.ClientHttpRequestFactorySettings.Redirects; +import org.springframework.boot.http.client.ClientHttpRequestFactorySettings; /** * {@link ConfigurationProperties @ConfigurationProperties} for a Spring's blocking HTTP @@ -29,131 +25,9 @@ * * @author Phillip Webb * @since 3.4.0 + * @see ClientHttpRequestFactorySettings */ @ConfigurationProperties("spring.http.client") -public class HttpClientProperties { - - /** - * Default factory used for a client HTTP request. - */ - private Factory factory; - - /** - * Handling for HTTP redirects. - */ - private Redirects redirects = Redirects.FOLLOW_WHEN_POSSIBLE; - - /** - * Default connect timeout for a client HTTP request. - */ - private Duration connectTimeout; - - /** - * Default read timeout for a client HTTP request. - */ - private Duration readTimeout; - - /** - * Default SSL configuration for a client HTTP request. - */ - private final Ssl ssl = new Ssl(); - - public Factory getFactory() { - return this.factory; - } - - public void setFactory(Factory factory) { - this.factory = factory; - } - - public Redirects getRedirects() { - return this.redirects; - } - - public void setRedirects(Redirects redirects) { - this.redirects = redirects; - } - - public Duration getConnectTimeout() { - return this.connectTimeout; - } - - public void setConnectTimeout(Duration connectTimeout) { - this.connectTimeout = connectTimeout; - } - - public Duration getReadTimeout() { - return this.readTimeout; - } - - public void setReadTimeout(Duration readTimeout) { - this.readTimeout = readTimeout; - } - - public Ssl getSsl() { - return this.ssl; - } - - /** - * Supported factory types. - */ - public enum Factory { - - /** - * Apache HttpComponents HttpClient. - */ - HTTP_COMPONENTS(ClientHttpRequestFactoryBuilder::httpComponents), - - /** - * Jetty's HttpClient. - */ - JETTY(ClientHttpRequestFactoryBuilder::jetty), - - /** - * Reactor-Netty. - */ - REACTOR(ClientHttpRequestFactoryBuilder::reactor), - - /** - * Java's HttpClient. - */ - JDK(ClientHttpRequestFactoryBuilder::jdk), - - /** - * Standard JDK facilities. - */ - SIMPLE(ClientHttpRequestFactoryBuilder::simple); - - private final Supplier> builderSupplier; - - Factory(Supplier> builderSupplier) { - this.builderSupplier = builderSupplier; - } - - ClientHttpRequestFactoryBuilder builder() { - return this.builderSupplier.get(); - } - - } - - /** - * SSL configuration. - */ - public static class Ssl { - - /** - * SSL bundle to use. - */ - private String bundle; - - public String getBundle() { - return this.bundle; - } - - public void setBundle(String bundle) { - this.bundle = bundle; - } - - } +public class HttpClientProperties extends AbstractHttpRequestFactoryProperties { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/AbstractClientHttpConnectorProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/AbstractClientHttpConnectorProperties.java new file mode 100644 index 000000000000..d452f9b28ee0 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/AbstractClientHttpConnectorProperties.java @@ -0,0 +1,100 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.client.reactive; + +import java.util.function.Supplier; + +import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.autoconfigure.http.client.AbstractHttpClientProperties; +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.http.client.HttpClientSettings; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorSettings; +import org.springframework.boot.ssl.SslBundles; +import org.springframework.http.client.reactive.ClientHttpConnector; + +/** + * Base {@link ConfigurationProperties @ConfigurationProperties} for configuring a + * {@link ClientHttpConnector}. + * + * @author Phillip Webb + * @since 3.5.0 + * @see ClientHttpConnectorSettings + */ +public abstract class AbstractClientHttpConnectorProperties extends AbstractHttpClientProperties { + + /** + * Default connector used for a client HTTP request. + */ + private Connector connector; + + public Connector getConnector() { + return this.connector; + } + + public void setConnector(Connector connector) { + this.connector = connector; + } + + @Override + protected HttpClientSettings httpClientSettings(ObjectProvider sslBundles) { + return super.httpClientSettings(sslBundles); + } + + protected final ClientHttpConnectorBuilder connectorBuilder(ClassLoader classLoader) { + Connector connector = getConnector(); + return (connector != null) ? connector.builder() : ClientHttpConnectorBuilder.detect(classLoader); + } + + /** + * Supported factory types. + */ + public enum Connector { + + /** + * Reactor-Netty. + */ + REACTOR(ClientHttpConnectorBuilder::reactor), + + /** + * Jetty's HttpClient. + */ + JETTY(ClientHttpConnectorBuilder::jetty), + + /** + * Apache HttpComponents HttpClient. + */ + HTTP_COMPONENTS(ClientHttpConnectorBuilder::httpComponents), + + /** + * Java's HttpClient. + */ + JDK(ClientHttpConnectorBuilder::jdk); + + private final Supplier> builderSupplier; + + Connector(Supplier> builderSupplier) { + this.builderSupplier = builderSupplier; + } + + ClientHttpConnectorBuilder builder() { + return this.builderSupplier.get(); + } + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/ClientHttpConnectorAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/ClientHttpConnectorAutoConfiguration.java new file mode 100644 index 000000000000..ce797e03ec1d --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/ClientHttpConnectorAutoConfiguration.java @@ -0,0 +1,106 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.client.reactive; + +import java.util.List; + +import reactor.core.publisher.Mono; + +import org.springframework.beans.factory.BeanClassLoaderAware; +import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.autoconfigure.AutoConfiguration; +import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; +import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.boot.autoconfigure.reactor.netty.ReactorNettyConfigurations; +import org.springframework.boot.autoconfigure.ssl.SslAutoConfiguration; +import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.boot.http.client.HttpClientSettings; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorSettings; +import org.springframework.boot.ssl.SslBundles; +import org.springframework.boot.util.LambdaSafe; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Import; +import org.springframework.context.annotation.Lazy; +import org.springframework.http.client.reactive.ClientHttpConnector; + +/** + * {@link EnableAutoConfiguration Auto-configuration} for + * {@link ClientHttpConnectorBuilder} and {@link ClientHttpConnectorSettings}. + * + * @author Phillip Webb + * @since 3.5.0 + */ +@AutoConfiguration(after = SslAutoConfiguration.class) +@ConditionalOnClass({ ClientHttpConnector.class, Mono.class }) +@EnableConfigurationProperties(HttpReactiveClientSettingsProperties.class) +public class ClientHttpConnectorAutoConfiguration implements BeanClassLoaderAware { + + private ClassLoader beanClassLoader; + + @Override + public void setBeanClassLoader(ClassLoader classLoader) { + this.beanClassLoader = classLoader; + } + + @Bean + @ConditionalOnMissingBean + ClientHttpConnectorBuilder clientHttpConnectorBuilder( + HttpReactiveClientSettingsProperties httpReactiveClientSettingsProperties, + ObjectProvider> clientHttpConnectorBuilderCustomizers) { + ClientHttpConnectorBuilder builder = httpReactiveClientSettingsProperties + .connectorBuilder(this.beanClassLoader); + return customize(builder, clientHttpConnectorBuilderCustomizers.orderedStream().toList()); + } + + @SuppressWarnings("unchecked") + private ClientHttpConnectorBuilder customize(ClientHttpConnectorBuilder builder, + List> customizers) { + ClientHttpConnectorBuilder[] builderReference = { builder }; + LambdaSafe.callbacks(ClientHttpConnectorBuilderCustomizer.class, customizers, builderReference[0]) + .invoke((customizer) -> builderReference[0] = customizer.customize(builderReference[0])); + return builderReference[0]; + } + + @Bean + @ConditionalOnMissingBean + ClientHttpConnectorSettings clientHttpConnectorSettings( + HttpReactiveClientSettingsProperties httpReactiveClientSettingsProperties, + ObjectProvider sslBundles) { + HttpClientSettings settings = httpReactiveClientSettingsProperties.httpClientSettings(sslBundles); + return new ClientHttpConnectorSettings(settings.redirects(), settings.connectTimeout(), settings.readTimeout(), + settings.sslBundle()); + } + + @Bean + @Lazy + @ConditionalOnMissingBean + ClientHttpConnector clientHttpConnector(ClientHttpConnectorBuilder clientHttpConnectorBuilder, + ClientHttpConnectorSettings clientHttpRequestFactorySettings) { + return clientHttpConnectorBuilder.build(clientHttpRequestFactorySettings); + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnClass(reactor.netty.http.client.HttpClient.class) + @Import(ReactorNettyConfigurations.ReactorResourceFactoryConfiguration.class) + static class ReactorNetty { + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/ClientHttpConnectorBuilderCustomizer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/ClientHttpConnectorBuilderCustomizer.java new file mode 100644 index 000000000000..67fd64e84dc6 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/ClientHttpConnectorBuilderCustomizer.java @@ -0,0 +1,38 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.client.reactive; + +import org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder; + +/** + * Customizer that can be used to modify the auto-configured + * {@link ClientHttpConnectorBuilder} when its type matches. + * + * @param the builder type + * @author Phillip Webb + * @since 3.5.0 + */ +public interface ClientHttpConnectorBuilderCustomizer> { + + /** + * Customize the given builder. + * @param builder the builder to customize + * @return the customized builder + */ + B customize(B builder); + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/HttpReactiveClientSettingsProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/HttpReactiveClientSettingsProperties.java new file mode 100644 index 000000000000..c9e3074179af --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/HttpReactiveClientSettingsProperties.java @@ -0,0 +1,33 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.client.reactive; + +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorSettings; + +/** + * {@link ConfigurationProperties @ConfigurationProperties} to configure settings that + * apply to Spring's reactive client HTTP connectors. + * + * @author Phillip Webb + * @since 3.5.0 + * @see ClientHttpConnectorSettings + */ +@ConfigurationProperties("spring.http.reactiveclient.settings") +public class HttpReactiveClientSettingsProperties extends AbstractClientHttpConnectorProperties { + +} diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/war/src/main/java/org/test/SampleApplication.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/package-info.java similarity index 75% rename from spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/war/src/main/java/org/test/SampleApplication.java rename to spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/package-info.java index 16c76e92c504..e60481c7c75a 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/war/src/main/java/org/test/SampleApplication.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/client/reactive/package-info.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -14,11 +14,7 @@ * limitations under the License. */ -package org.test; - -public class SampleApplication { - - public static void main(String[] args) { - } - -} +/** + * Auto-configuration for client-side reactive HTTP. + */ +package org.springframework.boot.autoconfigure.http.client.reactive; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/codec/CodecsAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/codec/CodecsAutoConfiguration.java index b8f703897107..3ed75eb911fc 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/codec/CodecsAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/codec/CodecsAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,7 +20,6 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; -import org.springframework.boot.autoconfigure.codec.CodecProperties; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.jackson.JacksonAutoConfiguration; @@ -29,7 +28,9 @@ import org.springframework.boot.web.codec.CodecCustomizer; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.core.Ordered; import org.springframework.core.annotation.Order; +import org.springframework.core.env.Environment; import org.springframework.http.codec.CodecConfigurer; import org.springframework.http.codec.json.Jackson2JsonDecoder; import org.springframework.http.codec.json.Jackson2JsonEncoder; @@ -47,7 +48,6 @@ */ @AutoConfiguration(after = JacksonAutoConfiguration.class) @ConditionalOnClass({ CodecConfigurer.class, WebClient.class }) -@EnableConfigurationProperties(CodecProperties.class) public class CodecsAutoConfiguration { private static final MimeType[] EMPTY_MIME_TYPES = {}; @@ -69,21 +69,48 @@ CodecCustomizer jacksonCodecCustomizer(ObjectMapper objectMapper) { } + @SuppressWarnings("removal") @Configuration(proxyBeanMethods = false) + @EnableConfigurationProperties({ org.springframework.boot.autoconfigure.codec.CodecProperties.class, + HttpCodecsProperties.class }) static class DefaultCodecsConfiguration { @Bean - @Order(0) - CodecCustomizer defaultCodecCustomizer(CodecProperties codecProperties) { - return (configurer) -> { + DefaultCodecCustomizer defaultCodecCustomizer( + org.springframework.boot.autoconfigure.codec.CodecProperties codecProperties, + HttpCodecsProperties httpCodecProperties, Environment environment) { + return new DefaultCodecCustomizer( + httpCodecProperties.isLogRequestDetails(codecProperties::isLogRequestDetails), + httpCodecProperties.getMaxInMemorySize(codecProperties::getMaxInMemorySize)); + } + + static final class DefaultCodecCustomizer implements CodecCustomizer, Ordered { + + private final boolean logRequestDetails; + + private final DataSize maxInMemorySize; + + DefaultCodecCustomizer(boolean logRequestDetails, DataSize maxInMemorySize) { + this.logRequestDetails = logRequestDetails; + this.maxInMemorySize = maxInMemorySize; + } + + @Override + public void customize(CodecConfigurer configurer) { PropertyMapper map = PropertyMapper.get(); CodecConfigurer.DefaultCodecs defaultCodecs = configurer.defaultCodecs(); - defaultCodecs.enableLoggingRequestDetails(codecProperties.isLogRequestDetails()); - map.from(codecProperties.getMaxInMemorySize()) + defaultCodecs.enableLoggingRequestDetails(this.logRequestDetails); + map.from(this.maxInMemorySize) .whenNonNull() .asInt(DataSize::toBytes) .to(defaultCodecs::maxInMemorySize); - }; + } + + @Override + public int getOrder() { + return 0; + } + } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/codec/HttpCodecsProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/codec/HttpCodecsProperties.java new file mode 100644 index 000000000000..2320016a7ee4 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/http/codec/HttpCodecsProperties.java @@ -0,0 +1,79 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.codec; + +import java.util.function.Supplier; + +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.util.unit.DataSize; + +/** + * {@link ConfigurationProperties Properties} for reactive HTTP codecs. + * + * @author Brian Clozel + * @author Andy Wilkinson + * @since 3.5.0 + */ +@ConfigurationProperties("spring.http.codecs") +public class HttpCodecsProperties { + + /** + * Whether to log form data at DEBUG level, and headers at TRACE level. + */ + private boolean logRequestDetails; + + @Deprecated(since = "3.5.0", forRemoval = true) + private boolean logRequestDetailsBound = false; + + /** + * Limit on the number of bytes that can be buffered whenever the input stream needs + * to be aggregated. This applies only to the auto-configured WebFlux server and + * WebClient instances. By default this is not set, in which case individual codec + * defaults apply. Most codecs are limited to 256K by default. + */ + private DataSize maxInMemorySize; + + @Deprecated(since = "3.5.0", forRemoval = true) + private boolean maxInMemorySizeBound = false; + + public boolean isLogRequestDetails() { + return this.logRequestDetails; + } + + boolean isLogRequestDetails(Supplier fallback) { + return this.logRequestDetailsBound ? this.logRequestDetails : fallback.get(); + } + + public void setLogRequestDetails(boolean logRequestDetails) { + this.logRequestDetails = logRequestDetails; + this.logRequestDetailsBound = true; + } + + public DataSize getMaxInMemorySize() { + return this.maxInMemorySize; + } + + DataSize getMaxInMemorySize(Supplier fallback) { + return this.maxInMemorySizeBound ? this.maxInMemorySize : fallback.get(); + } + + public void setMaxInMemorySize(DataSize maxInMemorySize) { + this.maxInMemorySize = maxInMemorySize; + this.maxInMemorySizeBound = true; + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/info/ProjectInfoProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/info/ProjectInfoProperties.java index ad533bc58815..60cb06d20233 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/info/ProjectInfoProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/info/ProjectInfoProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Stephane Nicoll * @since 1.4.0 */ -@ConfigurationProperties(prefix = "spring.info") +@ConfigurationProperties("spring.info") public class ProjectInfoProperties { private final Build build = new Build(); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/IntegrationAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/IntegrationAutoConfiguration.java index ebe3e7852e58..0f19e1db2bf8 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/IntegrationAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/IntegrationAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,10 +24,12 @@ import io.rsocket.transport.netty.server.TcpServerTransport; import org.springframework.beans.factory.BeanFactory; +import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.AutoConfigureBefore; import org.springframework.boot.autoconfigure.condition.AnyNestedCondition; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; @@ -84,6 +86,7 @@ * @author Vedran Pavic * @author Madhura Bhave * @author Yong-Hyun Kim + * @author Yanming Zhou * @since 1.1.0 */ @AutoConfiguration(after = { DataSourceAutoConfiguration.class, JmxAutoConfiguration.class, @@ -129,7 +132,8 @@ protected static class IntegrationConfiguration { @Bean(PollerMetadata.DEFAULT_POLLER) @ConditionalOnMissingBean(name = PollerMetadata.DEFAULT_POLLER) - public PollerMetadata defaultPollerMetadata(IntegrationProperties integrationProperties) { + public PollerMetadata defaultPollerMetadata(IntegrationProperties integrationProperties, + ObjectProvider customizers) { IntegrationProperties.Poller poller = integrationProperties.getPoller(); MutuallyExclusiveConfigurationPropertiesException.throwIfMultipleNonNullValuesIn((entries) -> { entries.put("spring.integration.poller.cron", @@ -142,6 +146,7 @@ public PollerMetadata defaultPollerMetadata(IntegrationProperties integrationPro map.from(poller::getMaxMessagesPerPoll).to(pollerMetadata::setMaxMessagesPerPoll); map.from(poller::getReceiveTimeout).as(Duration::toMillis).to(pollerMetadata::setReceiveTimeout); map.from(poller).as(this::asTrigger).to(pollerMetadata::setTrigger); + customizers.orderedStream().forEach((customizer) -> customizer.customize(pollerMetadata)); return pollerMetadata; } @@ -204,7 +209,7 @@ public SimpleAsyncTaskScheduler taskSchedulerVirtualThreads( @ConditionalOnClass(EnableIntegrationMBeanExport.class) @ConditionalOnMissingBean(value = IntegrationMBeanExporter.class, search = SearchStrategy.CURRENT) @ConditionalOnBean(MBeanServer.class) - @ConditionalOnProperty(prefix = "spring.jmx", name = "enabled", havingValue = "true", matchIfMissing = true) + @ConditionalOnBooleanProperty("spring.jmx.enabled") protected static class IntegrationJmxConfiguration { @Bean @@ -347,12 +352,13 @@ static class RemoteRSocketServerAddressConfigured extends AnyNestedCondition { super(ConfigurationPhase.REGISTER_BEAN); } - @ConditionalOnProperty(prefix = "spring.integration.rsocket.client", name = "uri") + @ConditionalOnProperty("spring.integration.rsocket.client.uri") static class WebSocketAddressConfigured { } - @ConditionalOnProperty(prefix = "spring.integration.rsocket.client", name = { "host", "port" }) + @ConditionalOnProperty({ "spring.integration.rsocket.client.host", + "spring.integration.rsocket.client.port" }) static class TcpAddressConfigured { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/IntegrationProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/IntegrationProperties.java index 5b7d2bbf764d..a7c6fdd4babe 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/IntegrationProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/IntegrationProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ * @author Artem Bilan * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.integration") +@ConfigurationProperties("spring.integration") public class IntegrationProperties { private final Channel channel = new Channel(); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/PollerMetadataCustomizer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/PollerMetadataCustomizer.java new file mode 100644 index 000000000000..d669faa5b991 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/integration/PollerMetadataCustomizer.java @@ -0,0 +1,37 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.integration; + +import org.springframework.integration.scheduling.PollerMetadata; + +/** + * Callback interface that can be implemented by beans wishing to customize the + * {@link PollerMetadata} whilst retaining default auto-configuration. + * + * @author Yanming Zhou + * @since 3.5.0 + */ +@FunctionalInterface +public interface PollerMetadataCustomizer { + + /** + * Customize the {@link PollerMetadata}. + * @param pollerMetadata the {@code PollerMetadata} to customize + */ + void customize(PollerMetadata pollerMetadata); + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jackson/JacksonAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jackson/JacksonAutoConfiguration.java index e388f4c3e5c3..cd21a090b39e 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jackson/JacksonAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jackson/JacksonAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -292,7 +292,7 @@ private void configurePropertyNamingStrategyField(Jackson2ObjectMapperBuilder bu // Find the field (this way we automatically support new constants // that may be added by Jackson in the future) Field field = findPropertyNamingStrategyField(fieldName); - Assert.notNull(field, () -> "Constant named '" + fieldName + "' not found"); + Assert.state(field != null, () -> "Constant named '" + fieldName + "' not found"); try { builder.propertyNamingStrategy((PropertyNamingStrategy) field.get(null)); } @@ -307,7 +307,7 @@ private Field findPropertyNamingStrategyField(String fieldName) { } private void configureModules(Jackson2ObjectMapperBuilder builder) { - builder.modulesToInstall(this.modules.toArray(new Module[0])); + builder.modulesToInstall((modules) -> modules.addAll(this.modules)); } private void configureLocale(Jackson2ObjectMapperBuilder builder) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jackson/JacksonProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jackson/JacksonProperties.java index 46162768d4f5..342bb1947cb4 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jackson/JacksonProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jackson/JacksonProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,7 +43,7 @@ * @author Eddú Meléndez * @since 1.2.0 */ -@ConfigurationProperties(prefix = "spring.jackson") +@ConfigurationProperties("spring.jackson") public class JacksonProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceAutoConfiguration.java index 0f85ec11034b..5fd4bda9e30d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,6 +23,7 @@ import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.AnyNestedCondition; import org.springframework.boot.autoconfigure.condition.ConditionMessage; +import org.springframework.boot.autoconfigure.condition.ConditionMessage.Style; import org.springframework.boot.autoconfigure.condition.ConditionOutcome; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; @@ -42,6 +43,7 @@ import org.springframework.core.env.Environment; import org.springframework.core.type.AnnotatedTypeMetadata; import org.springframework.jdbc.datasource.embedded.EmbeddedDatabaseType; +import org.springframework.util.ClassUtils; import org.springframework.util.StringUtils; /** @@ -95,7 +97,7 @@ static class PooledDataSourceCondition extends AnyNestedCondition { super(ConfigurationPhase.PARSE_CONFIGURATION); } - @ConditionalOnProperty(prefix = "spring.datasource", name = "type") + @ConditionalOnProperty("spring.datasource.type") static class ExplicitType { } @@ -132,6 +134,8 @@ static class EmbeddedDatabaseCondition extends SpringBootCondition { private static final String DATASOURCE_URL_PROPERTY = "spring.datasource.url"; + private static final String EMBEDDED_DATABASE_TYPE = "org.springframework.jdbc.datasource.embedded.EmbeddedDatabaseType"; + private final SpringBootCondition pooledCondition = new PooledDataSourceCondition(); @Override @@ -143,6 +147,10 @@ public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeM if (anyMatches(context, metadata, this.pooledCondition)) { return ConditionOutcome.noMatch(message.foundExactly("supported pooled data source")); } + if (!ClassUtils.isPresent(EMBEDDED_DATABASE_TYPE, context.getClassLoader())) { + return ConditionOutcome + .noMatch(message.didNotFind("required class").items(Style.QUOTE, EMBEDDED_DATABASE_TYPE)); + } EmbeddedDatabaseType type = EmbeddedDatabaseConnection.get(context.getClassLoader()).getType(); if (type == null) { return ConditionOutcome.noMatch(message.didNotFind("embedded database").atAll()); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceConfiguration.java index 1cebf37cd5e0..def27d6fccc0 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,6 +33,7 @@ import org.springframework.boot.jdbc.DatabaseDriver; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.core.env.Environment; import org.springframework.util.StringUtils; /** @@ -50,10 +51,17 @@ abstract class DataSourceConfiguration { @SuppressWarnings("unchecked") private static T createDataSource(JdbcConnectionDetails connectionDetails, Class type, ClassLoader classLoader) { - return (T) DataSourceBuilder.create(classLoader) - .type(type) - .driverClassName(connectionDetails.getDriverClassName()) - .url(connectionDetails.getJdbcUrl()) + return createDataSource(connectionDetails, type, classLoader, true); + } + + @SuppressWarnings("unchecked") + private static T createDataSource(JdbcConnectionDetails connectionDetails, Class type, + ClassLoader classLoader, boolean applyDriverClassName) { + DataSourceBuilder builder = DataSourceBuilder.create(classLoader).type(type); + if (applyDriverClassName) { + builder.driverClassName(connectionDetails.getDriverClassName()); + } + return (T) builder.url(connectionDetails.getJdbcUrl()) .username(connectionDetails.getUsername()) .password(connectionDetails.getPassword()) .build(); @@ -77,7 +85,7 @@ static TomcatJdbcConnectionDetailsBeanPostProcessor tomcatJdbcConnectionDetailsB } @Bean - @ConfigurationProperties(prefix = "spring.datasource.tomcat") + @ConfigurationProperties("spring.datasource.tomcat") org.apache.tomcat.jdbc.pool.DataSource dataSource(DataSourceProperties properties, JdbcConnectionDetails connectionDetails) { Class dataSourceType = org.apache.tomcat.jdbc.pool.DataSource.class; @@ -112,10 +120,12 @@ static HikariJdbcConnectionDetailsBeanPostProcessor jdbcConnectionDetailsHikariB } @Bean - @ConfigurationProperties(prefix = "spring.datasource.hikari") - HikariDataSource dataSource(DataSourceProperties properties, JdbcConnectionDetails connectionDetails) { + @ConfigurationProperties("spring.datasource.hikari") + HikariDataSource dataSource(DataSourceProperties properties, JdbcConnectionDetails connectionDetails, + Environment environment) { + String dataSourceClassName = environment.getProperty("spring.datasource.hikari.data-source-class-name"); HikariDataSource dataSource = createDataSource(connectionDetails, HikariDataSource.class, - properties.getClassLoader()); + properties.getClassLoader(), dataSourceClassName == null); if (StringUtils.hasText(properties.getName())) { dataSource.setPoolName(properties.getName()); } @@ -141,7 +151,7 @@ static Dbcp2JdbcConnectionDetailsBeanPostProcessor dbcp2JdbcConnectionDetailsBea } @Bean - @ConfigurationProperties(prefix = "spring.datasource.dbcp2") + @ConfigurationProperties("spring.datasource.dbcp2") org.apache.commons.dbcp2.BasicDataSource dataSource(DataSourceProperties properties, JdbcConnectionDetails connectionDetails) { Class dataSourceType = org.apache.commons.dbcp2.BasicDataSource.class; @@ -167,7 +177,7 @@ static OracleUcpJdbcConnectionDetailsBeanPostProcessor oracleUcpJdbcConnectionDe } @Bean - @ConfigurationProperties(prefix = "spring.datasource.oracleucp") + @ConfigurationProperties("spring.datasource.oracleucp") PoolDataSourceImpl dataSource(DataSourceProperties properties, JdbcConnectionDetails connectionDetails) throws SQLException { PoolDataSourceImpl dataSource = createDataSource(connectionDetails, PoolDataSourceImpl.class, diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceJmxConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceJmxConfiguration.java index e07b2f1f2f99..2dfce930e2e7 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceJmxConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceJmxConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,9 +28,9 @@ import org.apache.tomcat.jdbc.pool.PoolConfiguration; import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnSingleCandidate; import org.springframework.boot.jdbc.DataSourceUnwrapper; import org.springframework.context.annotation.Bean; @@ -43,7 +43,7 @@ * @author Stephane Nicoll */ @Configuration(proxyBeanMethods = false) -@ConditionalOnProperty(prefix = "spring.jmx", name = "enabled", havingValue = "true", matchIfMissing = true) +@ConditionalOnBooleanProperty("spring.jmx.enabled") class DataSourceJmxConfiguration { private static final Log logger = LogFactory.getLog(DataSourceJmxConfiguration.class); @@ -74,7 +74,7 @@ private void validateMBeans() { } @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.datasource.tomcat", name = "jmx-enabled") + @ConditionalOnBooleanProperty("spring.datasource.tomcat.jmx-enabled") @ConditionalOnClass(DataSourceProxy.class) @ConditionalOnSingleCandidate(DataSource.class) static class TomcatDataSourceJmxConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceProperties.java index 1504b86b83ef..7af946e97cb5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -44,7 +44,7 @@ * @author Scott Frederick * @since 1.1.0 */ -@ConfigurationProperties(prefix = "spring.datasource") +@ConfigurationProperties("spring.datasource") public class DataSourceProperties implements BeanClassLoaderAware, InitializingBean { private ClassLoader classLoader; @@ -171,6 +171,15 @@ public void setDriverClassName(String driverClassName) { * @since 1.4.0 */ public String determineDriverClassName() { + String driverClassName = findDriverClassName(); + if (!StringUtils.hasText(driverClassName)) { + throw new DataSourceBeanCreationException("Failed to determine a suitable driver class", this, + this.embeddedDatabaseConnection); + } + return driverClassName; + } + + String findDriverClassName() { if (StringUtils.hasText(this.driverClassName)) { Assert.state(driverClassIsLoadable(), () -> "Cannot load driver class: " + this.driverClassName); return this.driverClassName; @@ -182,10 +191,6 @@ public String determineDriverClassName() { if (!StringUtils.hasText(driverClassName)) { driverClassName = this.embeddedDatabaseConnection.getDriverClassName(); } - if (!StringUtils.hasText(driverClassName)) { - throw new DataSourceBeanCreationException("Failed to determine a suitable driver class", this, - this.embeddedDatabaseConnection); - } return driverClassName; } @@ -277,7 +282,7 @@ public String determineUsername() { if (StringUtils.hasText(this.username)) { return this.username; } - if (EmbeddedDatabaseConnection.isEmbedded(determineDriverClassName(), determineUrl())) { + if (EmbeddedDatabaseConnection.isEmbedded(findDriverClassName(), determineUrl())) { return "sa"; } return null; @@ -305,7 +310,7 @@ public String determinePassword() { if (StringUtils.hasText(this.password)) { return this.password; } - if (EmbeddedDatabaseConnection.isEmbedded(determineDriverClassName(), determineUrl())) { + if (EmbeddedDatabaseConnection.isEmbedded(findDriverClassName(), determineUrl())) { return ""; } return null; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceTransactionManagerAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceTransactionManagerAutoConfiguration.java index b8b9615e2ccb..27cc702e0d03 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceTransactionManagerAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/DataSourceTransactionManagerAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,7 +28,6 @@ import org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration; import org.springframework.boot.autoconfigure.transaction.TransactionManagerCustomizationAutoConfiguration; import org.springframework.boot.autoconfigure.transaction.TransactionManagerCustomizers; -import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.core.Ordered; @@ -48,10 +47,9 @@ * @since 1.0.0 */ @AutoConfiguration(before = TransactionAutoConfiguration.class, - after = TransactionManagerCustomizationAutoConfiguration.class) + after = { DataSourceAutoConfiguration.class, TransactionManagerCustomizationAutoConfiguration.class }) @ConditionalOnClass({ DataSource.class, JdbcTemplate.class, TransactionManager.class }) @AutoConfigureOrder(Ordered.LOWEST_PRECEDENCE) -@EnableConfigurationProperties(DataSourceProperties.class) public class DataSourceTransactionManagerAutoConfiguration { @Configuration(proxyBeanMethods = false) diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/HikariJdbcConnectionDetailsBeanPostProcessor.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/HikariJdbcConnectionDetailsBeanPostProcessor.java index ca4b1b18f6a4..3f05a8ffb8f5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/HikariJdbcConnectionDetailsBeanPostProcessor.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/HikariJdbcConnectionDetailsBeanPostProcessor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,10 @@ protected Object processDataSource(HikariDataSource dataSource, JdbcConnectionDe dataSource.setJdbcUrl(connectionDetails.getJdbcUrl()); dataSource.setUsername(connectionDetails.getUsername()); dataSource.setPassword(connectionDetails.getPassword()); - dataSource.setDriverClassName(connectionDetails.getDriverClassName()); + String driverClassName = connectionDetails.getDriverClassName(); + if (driverClassName != null) { + dataSource.setDriverClassName(driverClassName); + } return dataSource; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JdbcProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JdbcProperties.java index 0e0d3560ec56..6c9a1ab412c0 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JdbcProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JdbcProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Stephane Nicoll * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.jdbc") +@ConfigurationProperties("spring.jdbc") public class JdbcProperties { private final Template template = new Template(); @@ -43,6 +43,12 @@ public Template getTemplate() { */ public static class Template { + /** + * Whether to ignore JDBC statement warnings (SQLWarning). When set to false, + * throw an SQLWarningException instead. + */ + private boolean ignoreWarnings = true; + /** * Number of rows that should be fetched from the database when more rows are * needed. Use -1 to use the JDBC driver's default configuration. @@ -61,6 +67,31 @@ public static class Template { @DurationUnit(ChronoUnit.SECONDS) private Duration queryTimeout; + /** + * Whether results processing should be skipped. Can be used to optimize callable + * statement processing when we know that no results are being passed back. + */ + private boolean skipResultsProcessing; + + /** + * Whether undeclared results should be skipped. + */ + private boolean skipUndeclaredResults; + + /** + * Whether execution of a CallableStatement will return the results in a Map that + * uses case-insensitive names for the parameters. + */ + private boolean resultsMapCaseInsensitive; + + public boolean isIgnoreWarnings() { + return this.ignoreWarnings; + } + + public void setIgnoreWarnings(boolean ignoreWarnings) { + this.ignoreWarnings = ignoreWarnings; + } + public int getFetchSize() { return this.fetchSize; } @@ -85,6 +116,30 @@ public void setQueryTimeout(Duration queryTimeout) { this.queryTimeout = queryTimeout; } + public boolean isSkipResultsProcessing() { + return this.skipResultsProcessing; + } + + public void setSkipResultsProcessing(boolean skipResultsProcessing) { + this.skipResultsProcessing = skipResultsProcessing; + } + + public boolean isSkipUndeclaredResults() { + return this.skipUndeclaredResults; + } + + public void setSkipUndeclaredResults(boolean skipUndeclaredResults) { + this.skipUndeclaredResults = skipUndeclaredResults; + } + + public boolean isResultsMapCaseInsensitive() { + return this.resultsMapCaseInsensitive; + } + + public void setResultsMapCaseInsensitive(boolean resultsMapCaseInsensitive) { + this.resultsMapCaseInsensitive = resultsMapCaseInsensitive; + } + } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JdbcTemplateConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JdbcTemplateConfiguration.java index 2043304293ab..86a584c19a5b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JdbcTemplateConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JdbcTemplateConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,17 +18,20 @@ import javax.sql.DataSource; +import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Primary; import org.springframework.jdbc.core.JdbcOperations; import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.support.SQLExceptionTranslator; /** * Configuration for {@link JdbcTemplateConfiguration}. * * @author Stephane Nicoll + * @author Yanming Zhou */ @Configuration(proxyBeanMethods = false) @ConditionalOnMissingBean(JdbcOperations.class) @@ -36,14 +39,20 @@ class JdbcTemplateConfiguration { @Bean @Primary - JdbcTemplate jdbcTemplate(DataSource dataSource, JdbcProperties properties) { + JdbcTemplate jdbcTemplate(DataSource dataSource, JdbcProperties properties, + ObjectProvider sqlExceptionTranslator) { JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource); JdbcProperties.Template template = properties.getTemplate(); + jdbcTemplate.setIgnoreWarnings(template.isIgnoreWarnings()); jdbcTemplate.setFetchSize(template.getFetchSize()); jdbcTemplate.setMaxRows(template.getMaxRows()); if (template.getQueryTimeout() != null) { jdbcTemplate.setQueryTimeout((int) template.getQueryTimeout().getSeconds()); } + jdbcTemplate.setSkipResultsProcessing(template.isSkipResultsProcessing()); + jdbcTemplate.setSkipUndeclaredResults(template.isSkipUndeclaredResults()); + jdbcTemplate.setResultsMapCaseInsensitive(template.isResultsMapCaseInsensitive()); + sqlExceptionTranslator.ifUnique(jdbcTemplate::setExceptionTranslator); return jdbcTemplate; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JndiDataSourceAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JndiDataSourceAutoConfiguration.java index 199193af53f0..8e7ae07b72db 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JndiDataSourceAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/JndiDataSourceAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,7 @@ */ @AutoConfiguration(before = { XADataSourceAutoConfiguration.class, DataSourceAutoConfiguration.class }) @ConditionalOnClass({ DataSource.class, EmbeddedDatabaseType.class }) -@ConditionalOnProperty(prefix = "spring.datasource", name = "jndi-name") +@ConditionalOnProperty("spring.datasource.jndi-name") @EnableConfigurationProperties(DataSourceProperties.class) public class JndiDataSourceAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/PropertiesJdbcConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/PropertiesJdbcConnectionDetails.java index 40109d89aa9b..6310e8d8f3c1 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/PropertiesJdbcConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/PropertiesJdbcConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/XADataSourceAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/XADataSourceAutoConfiguration.java index b78ec02c7f45..5eff8fadae99 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/XADataSourceAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jdbc/XADataSourceAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -96,7 +96,8 @@ private XADataSource createXaDataSourceInstance(String className) { try { Class dataSourceClass = ClassUtils.forName(className, this.classLoader); Object instance = BeanUtils.instantiateClass(dataSourceClass); - Assert.isInstanceOf(XADataSource.class, instance); + Assert.state(instance instanceof XADataSource, + () -> "DataSource class " + className + " is not an XADataSource"); return (XADataSource) instance; } catch (Exception ex) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jersey/JerseyAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jersey/JerseyAutoConfiguration.java index be0d886e57e9..a3d86633e08e 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jersey/JerseyAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jersey/JerseyAutoConfiguration.java @@ -115,7 +115,7 @@ public JerseyApplicationPath jerseyApplicationPath() { @Bean @ConditionalOnMissingBean(name = "jerseyFilterRegistration") - @ConditionalOnProperty(prefix = "spring.jersey", name = "type", havingValue = "filter") + @ConditionalOnProperty(name = "spring.jersey.type", havingValue = "filter") public FilterRegistrationBean jerseyFilterRegistration(JerseyApplicationPath applicationPath) { FilterRegistrationBean registration = new FilterRegistrationBean<>(); registration.setFilter(new ServletContainer(this.config)); @@ -137,7 +137,7 @@ private String stripPattern(String path) { @Bean @ConditionalOnMissingBean(name = "jerseyServletRegistration") - @ConditionalOnProperty(prefix = "spring.jersey", name = "type", havingValue = "servlet", matchIfMissing = true) + @ConditionalOnProperty(name = "spring.jersey.type", havingValue = "servlet", matchIfMissing = true) public ServletRegistrationBean jerseyServletRegistration(JerseyApplicationPath applicationPath) { ServletRegistrationBean registration = new ServletRegistrationBean<>( new ServletContainer(this.config), applicationPath.getUrlMapping()); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jersey/JerseyProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jersey/JerseyProperties.java index c69cf2d53009..06ca49053e5a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jersey/JerseyProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jersey/JerseyProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Stephane Nicoll * @since 1.2.0 */ -@ConfigurationProperties(prefix = "spring.jersey") +@ConfigurationProperties("spring.jersey") public class JerseyProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/DefaultJmsListenerContainerFactoryConfigurer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/DefaultJmsListenerContainerFactoryConfigurer.java index f1165151eb27..098c0b2c048b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/DefaultJmsListenerContainerFactoryConfigurer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/DefaultJmsListenerContainerFactoryConfigurer.java @@ -101,7 +101,7 @@ void setJmsProperties(JmsProperties jmsProperties) { * Set the {@link ObservationRegistry} to use. * @param observationRegistry the {@link ObservationRegistry} * @since 3.2.1 - * @deprecated since 3.3.10 for removal in 3.6.0 as this should have been package + * @deprecated since 3.3.10 for removal in 4.0.0 as this should have been package * private */ @Deprecated(since = "3.3.10", forRemoval = true) @@ -116,8 +116,8 @@ public void setObservationRegistry(ObservationRegistry observationRegistry) { * @param connectionFactory the {@link ConnectionFactory} to use */ public void configure(DefaultJmsListenerContainerFactory factory, ConnectionFactory connectionFactory) { - Assert.notNull(factory, "Factory must not be null"); - Assert.notNull(connectionFactory, "ConnectionFactory must not be null"); + Assert.notNull(factory, "'factory' must not be null"); + Assert.notNull(connectionFactory, "'connectionFactory' must not be null"); JmsProperties.Listener listenerProperties = this.jmsProperties.getListener(); Session sessionProperties = listenerProperties.getSession(); factory.setConnectionFactory(connectionFactory); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/JmsProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/JmsProperties.java index 126ffd065f31..b64b2a258ab9 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/JmsProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/JmsProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ * @author Vedran Pavic * @since 1.0.0 */ -@ConfigurationProperties(prefix = "spring.jms") +@ConfigurationProperties("spring.jms") public class JmsProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/JndiConnectionFactoryAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/JndiConnectionFactoryAutoConfiguration.java index 9d413a087dee..0475a4fe3b10 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/JndiConnectionFactoryAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/JndiConnectionFactoryAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -89,7 +89,7 @@ static class Jndi { } - @ConditionalOnProperty(prefix = "spring.jms", name = "jndi-name") + @ConditionalOnProperty("spring.jms.jndi-name") static class Property { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQConnectionFactoryConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQConnectionFactoryConfiguration.java index 897c69576c0d..01a3c82825fd 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQConnectionFactoryConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQConnectionFactoryConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,9 +22,9 @@ import org.messaginghub.pooled.jms.JmsPoolConnectionFactory; import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.jms.JmsPoolConnectionFactoryFactory; import org.springframework.boot.autoconfigure.jms.JmsProperties; import org.springframework.context.annotation.Bean; @@ -46,12 +46,11 @@ class ActiveMQConnectionFactoryConfiguration { @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.activemq.pool", name = "enabled", havingValue = "false", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.activemq.pool.enabled", havingValue = false, matchIfMissing = true) static class SimpleConnectionFactoryConfiguration { @Bean - @ConditionalOnProperty(prefix = "spring.jms.cache", name = "enabled", havingValue = "false") + @ConditionalOnBooleanProperty(name = "spring.jms.cache.enabled", havingValue = false) ActiveMQConnectionFactory jmsConnectionFactory(ActiveMQProperties properties, ObjectProvider factoryCustomizers, ActiveMQConnectionDetails connectionDetails) { @@ -70,8 +69,7 @@ private static ActiveMQConnectionFactory createJmsConnectionFactory(ActiveMQProp @Configuration(proxyBeanMethods = false) @ConditionalOnClass(CachingConnectionFactory.class) - @ConditionalOnProperty(prefix = "spring.jms.cache", name = "enabled", havingValue = "true", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.jms.cache.enabled", matchIfMissing = true) static class CachingConnectionFactoryConfiguration { @Bean @@ -96,7 +94,7 @@ CachingConnectionFactory jmsConnectionFactory(JmsProperties jmsProperties, Activ static class PooledConnectionFactoryConfiguration { @Bean(destroyMethod = "stop") - @ConditionalOnProperty(prefix = "spring.activemq.pool", name = "enabled", havingValue = "true") + @ConditionalOnBooleanProperty("spring.activemq.pool.enabled") JmsPoolConnectionFactory jmsConnectionFactory(ActiveMQProperties properties, ObjectProvider factoryCustomizers, ActiveMQConnectionDetails connectionDetails) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQConnectionFactoryConfigurer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQConnectionFactoryConfigurer.java index aa9c027540f4..143ff15267ae 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQConnectionFactoryConfigurer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQConnectionFactoryConfigurer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,7 @@ class ActiveMQConnectionFactoryConfigurer { ActiveMQConnectionFactoryConfigurer(ActiveMQProperties properties, List factoryCustomizers) { - Assert.notNull(properties, "Properties must not be null"); + Assert.notNull(properties, "'properties' must not be null"); this.properties = properties; this.factoryCustomizers = (factoryCustomizers != null) ? factoryCustomizers : Collections.emptyList(); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQProperties.java index dc4bb98a2b61..b1cb52cc274f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * @author Eddú Meléndez * @since 3.1.0 */ -@ConfigurationProperties(prefix = "spring.activemq") +@ConfigurationProperties("spring.activemq") public class ActiveMQProperties { private static final String DEFAULT_EMBEDDED_BROKER_URL = "vm://localhost?broker.persistent=false"; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQXAConnectionFactoryConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQXAConnectionFactoryConfiguration.java index d03fabc1192a..62acb526d9e5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQXAConnectionFactoryConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/activemq/ActiveMQXAConnectionFactoryConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,9 +23,9 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.jms.XAConnectionFactoryWrapper; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -57,8 +57,7 @@ ConnectionFactory jmsConnectionFactory(ActiveMQProperties properties, } @Bean - @ConditionalOnProperty(prefix = "spring.activemq.pool", name = "enabled", havingValue = "false", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.activemq.pool.enabled", havingValue = false, matchIfMissing = true) ActiveMQConnectionFactory nonXaJmsConnectionFactory(ActiveMQProperties properties, ObjectProvider factoryCustomizers, ActiveMQConnectionDetails connectionDetails) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisConnectionFactoryConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisConnectionFactoryConfiguration.java index ab4e1f6939c4..5ffa5ae225ca 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisConnectionFactoryConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisConnectionFactoryConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,9 +22,9 @@ import org.messaginghub.pooled.jms.JmsPoolConnectionFactory; import org.springframework.beans.factory.ListableBeanFactory; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.jms.JmsPoolConnectionFactoryFactory; import org.springframework.boot.autoconfigure.jms.JmsProperties; import org.springframework.context.annotation.Bean; @@ -43,12 +43,11 @@ class ArtemisConnectionFactoryConfiguration { @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.artemis.pool", name = "enabled", havingValue = "false", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.artemis.pool.enabled", havingValue = false, matchIfMissing = true) static class SimpleConnectionFactoryConfiguration { @Bean(name = "jmsConnectionFactory") - @ConditionalOnProperty(prefix = "spring.jms.cache", name = "enabled", havingValue = "false") + @ConditionalOnBooleanProperty(name = "spring.jms.cache.enabled", havingValue = false) ActiveMQConnectionFactory jmsConnectionFactory(ArtemisProperties properties, ListableBeanFactory beanFactory, ArtemisConnectionDetails connectionDetails) { return createJmsConnectionFactory(properties, connectionDetails, beanFactory); @@ -62,8 +61,7 @@ private static ActiveMQConnectionFactory createJmsConnectionFactory(ArtemisPrope @Configuration(proxyBeanMethods = false) @ConditionalOnClass(CachingConnectionFactory.class) - @ConditionalOnProperty(prefix = "spring.jms.cache", name = "enabled", havingValue = "true", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.jms.cache.enabled", matchIfMissing = true) static class CachingConnectionFactoryConfiguration { @Bean(name = "jmsConnectionFactory") @@ -85,7 +83,7 @@ CachingConnectionFactory cachingJmsConnectionFactory(JmsProperties jmsProperties @Configuration(proxyBeanMethods = false) @ConditionalOnClass({ JmsPoolConnectionFactory.class, PooledObject.class }) - @ConditionalOnProperty(prefix = "spring.artemis.pool", name = "enabled", havingValue = "true") + @ConditionalOnBooleanProperty("spring.artemis.pool.enabled") static class PooledConnectionFactoryConfiguration { @Bean(destroyMethod = "stop") diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisConnectionFactoryFactory.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisConnectionFactoryFactory.java index 957c17cc4b84..5fc00df65e3c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisConnectionFactoryFactory.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisConnectionFactoryFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,9 +53,9 @@ class ArtemisConnectionFactoryFactory { ArtemisConnectionFactoryFactory(ListableBeanFactory beanFactory, ArtemisProperties properties, ArtemisConnectionDetails connectionDetails) { - Assert.notNull(beanFactory, "BeanFactory must not be null"); - Assert.notNull(properties, "Properties must not be null"); - Assert.notNull(connectionDetails, "ConnectionDetails must not be null"); + Assert.notNull(beanFactory, "'beanFactory' must not be null"); + Assert.notNull(properties, "'properties' must not be null"); + Assert.notNull(connectionDetails, "'connectionDetails' must not be null"); this.beanFactory = beanFactory; this.properties = properties; this.connectionDetails = connectionDetails; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisEmbeddedServerConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisEmbeddedServerConfiguration.java index a3d2128eade8..27dd2d4dd8b0 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisEmbeddedServerConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisEmbeddedServerConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,9 +28,9 @@ import org.apache.activemq.artemis.jms.server.config.impl.TopicConfigurationImpl; import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -43,8 +43,7 @@ */ @Configuration(proxyBeanMethods = false) @ConditionalOnClass(EmbeddedActiveMQ.class) -@ConditionalOnProperty(prefix = "spring.artemis.embedded", name = "enabled", havingValue = "true", - matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.artemis.embedded.enabled", matchIfMissing = true) class ArtemisEmbeddedServerConfiguration { private final ArtemisProperties properties; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisProperties.java index 5341b5ab710a..5447ededf25e 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jms/artemis/ArtemisProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ * @author Justin Bertram * @since 1.3.0 */ -@ConfigurationProperties(prefix = "spring.artemis") +@ConfigurationProperties("spring.artemis") public class ArtemisProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jmx/JmxAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jmx/JmxAutoConfiguration.java index e39fc0373ef1..fb762f62ca58 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jmx/JmxAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jmx/JmxAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,9 +21,9 @@ import org.springframework.beans.factory.BeanFactory; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.SearchStrategy; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; @@ -52,7 +52,7 @@ @AutoConfiguration @EnableConfigurationProperties(JmxProperties.class) @ConditionalOnClass({ MBeanExporter.class }) -@ConditionalOnProperty(prefix = "spring.jmx", name = "enabled", havingValue = "true") +@ConditionalOnBooleanProperty("spring.jmx.enabled") public class JmxAutoConfiguration { private final JmxProperties properties; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jmx/JmxProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jmx/JmxProperties.java index 57b10a74ff5b..afa5a3ef4998 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jmx/JmxProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jmx/JmxProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ * @author Scott Frederick * @since 2.7.0 */ -@ConfigurationProperties(prefix = "spring.jmx") +@ConfigurationProperties("spring.jmx") public class JmxProperties { /** @@ -53,7 +53,7 @@ public class JmxProperties { */ private RegistrationPolicy registrationPolicy = RegistrationPolicy.FAIL_ON_EXISTING; - public boolean getEnabled() { + public boolean isEnabled() { return this.enabled; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/DefaultExceptionTranslatorExecuteListener.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/DefaultExceptionTranslatorExecuteListener.java index 199a62dda0e0..c6d67c7a7c3a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/DefaultExceptionTranslatorExecuteListener.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/DefaultExceptionTranslatorExecuteListener.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -63,7 +63,7 @@ final class DefaultExceptionTranslatorExecuteListener implements ExceptionTransl private DefaultExceptionTranslatorExecuteListener(Log logger, Function translatorFactory) { - Assert.notNull(translatorFactory, "TranslatorFactory must not be null"); + Assert.notNull(translatorFactory, "'translatorFactory' must not be null"); this.logger = logger; this.translatorFactory = translatorFactory; } diff --git a/spring-boot-tests/spring-boot-smoke-tests/spring-boot-smoke-test-pulsar4/src/main/java/smoketest/pulsar/SampleMessage.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JaxbNotAvailableException.java similarity index 68% rename from spring-boot-tests/spring-boot-smoke-tests/spring-boot-smoke-test-pulsar4/src/main/java/smoketest/pulsar/SampleMessage.java rename to spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JaxbNotAvailableException.java index 3887ce61f13a..963264373165 100644 --- a/spring-boot-tests/spring-boot-smoke-tests/spring-boot-smoke-test-pulsar4/src/main/java/smoketest/pulsar/SampleMessage.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JaxbNotAvailableException.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -14,7 +14,13 @@ * limitations under the License. */ -package smoketest.pulsar; +package org.springframework.boot.autoconfigure.jooq; + +/** + * Exception to be thrown if JAXB is not available. + * + * @author Moritz Halbritter + */ +class JaxbNotAvailableException extends RuntimeException { -record SampleMessage(Integer id, String content) { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JaxbNotAvailableExceptionFailureAnalyzer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JaxbNotAvailableExceptionFailureAnalyzer.java new file mode 100644 index 000000000000..dfb9eb9892c2 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JaxbNotAvailableExceptionFailureAnalyzer.java @@ -0,0 +1,36 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.jooq; + +import org.springframework.boot.diagnostics.AbstractFailureAnalyzer; +import org.springframework.boot.diagnostics.FailureAnalysis; +import org.springframework.boot.diagnostics.FailureAnalyzer; + +/** + * {@link FailureAnalyzer} for {@link JaxbNotAvailableException}. + * + * @author Moritz Halbritter + */ +class JaxbNotAvailableExceptionFailureAnalyzer extends AbstractFailureAnalyzer { + + @Override + protected FailureAnalysis analyze(Throwable rootFailure, JaxbNotAvailableException cause) { + return new FailureAnalysis("Unable to unmarshal jOOQ settings because JAXB is not available.", + "Add JAXB to the classpath or remove the spring.jooq.config property.", cause); + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqAutoConfiguration.java index 6e0afdfa09a4..ffb020e29de9 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,16 +16,32 @@ package org.springframework.boot.autoconfigure.jooq; +import java.io.IOException; +import java.io.InputStream; + import javax.sql.DataSource; +import javax.xml.XMLConstants; +import javax.xml.parsers.ParserConfigurationException; +import javax.xml.parsers.SAXParser; +import javax.xml.parsers.SAXParserFactory; +import javax.xml.transform.Source; +import javax.xml.transform.sax.SAXSource; +import jakarta.xml.bind.JAXBContext; +import jakarta.xml.bind.JAXBException; import org.jooq.ConnectionProvider; import org.jooq.DSLContext; import org.jooq.ExecuteListenerProvider; import org.jooq.TransactionProvider; +import org.jooq.conf.Settings; import org.jooq.impl.DataSourceConnectionProvider; import org.jooq.impl.DefaultConfiguration; import org.jooq.impl.DefaultDSLContext; import org.jooq.impl.DefaultExecuteListenerProvider; +import org.xml.sax.InputSource; +import org.xml.sax.SAXException; +import org.xml.sax.SAXNotRecognizedException; +import org.xml.sax.SAXNotSupportedException; import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; @@ -33,20 +49,25 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; import org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; import org.springframework.core.annotation.Order; +import org.springframework.core.io.Resource; import org.springframework.jdbc.datasource.TransactionAwareDataSourceProxy; import org.springframework.transaction.PlatformTransactionManager; +import org.springframework.util.Assert; +import org.springframework.util.ClassUtils; /** - * {@link EnableAutoConfiguration Auto-configuration} for JOOQ. + * {@link EnableAutoConfiguration Auto-configuration} for jOOQ. * * @author Andreas Ahlenstorf * @author Michael Simons * @author Dmytro Nosan + * @author Moritz Halbritter * @since 1.3.0 */ @AutoConfiguration(after = { DataSourceAutoConfiguration.class, TransactionAutoConfiguration.class }) @@ -89,17 +110,65 @@ public DefaultDSLContext dslContext(org.jooq.Configuration configuration) { @Bean @ConditionalOnMissingBean(org.jooq.Configuration.class) - public DefaultConfiguration jooqConfiguration(JooqProperties properties, ConnectionProvider connectionProvider, + DefaultConfiguration jooqConfiguration(JooqProperties properties, ConnectionProvider connectionProvider, DataSource dataSource, ObjectProvider transactionProvider, ObjectProvider executeListenerProviders, - ObjectProvider configurationCustomizers) { + ObjectProvider configurationCustomizers, + ObjectProvider settingsProvider) { DefaultConfiguration configuration = new DefaultConfiguration(); configuration.set(properties.determineSqlDialect(dataSource)); configuration.set(connectionProvider); transactionProvider.ifAvailable(configuration::set); + settingsProvider.ifAvailable(configuration::set); configuration.set(executeListenerProviders.orderedStream().toArray(ExecuteListenerProvider[]::new)); configurationCustomizers.orderedStream().forEach((customizer) -> customizer.customize(configuration)); return configuration; } + @Bean + @ConditionalOnProperty("spring.jooq.config") + @ConditionalOnMissingBean(Settings.class) + Settings settings(JooqProperties properties) throws IOException { + if (!ClassUtils.isPresent("jakarta.xml.bind.JAXBContext", null)) { + throw new JaxbNotAvailableException(); + } + Resource resource = properties.getConfig(); + Assert.state(resource.exists(), + () -> "Resource %s set in spring.jooq.config does not exist".formatted(resource)); + try (InputStream stream = resource.getInputStream()) { + return new JaxbSettingsLoader().load(stream); + } + } + + /** + * Load {@link Settings} with + * XML External Entity Prevention. + */ + private static final class JaxbSettingsLoader { + + private Settings load(InputStream inputStream) { + try { + SAXParser parser = createParserFactory().newSAXParser(); + Source source = new SAXSource(parser.getXMLReader(), new InputSource(inputStream)); + JAXBContext context = JAXBContext.newInstance(Settings.class); + return context.createUnmarshaller().unmarshal(source, Settings.class).getValue(); + } + catch (ParserConfigurationException | JAXBException | SAXException ex) { + throw new IllegalStateException("Failed to unmarshal settings", ex); + } + } + + private SAXParserFactory createParserFactory() + throws ParserConfigurationException, SAXNotRecognizedException, SAXNotSupportedException { + SAXParserFactory factory = SAXParserFactory.newInstance(); + factory.setFeature("http://apache.org/xml/features/disallow-doctype-decl", true); + factory.setFeature(XMLConstants.FEATURE_SECURE_PROCESSING, true); + factory.setNamespaceAware(true); + factory.setXIncludeAware(false); + return factory; + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqExceptionTranslator.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqExceptionTranslator.java deleted file mode 100644 index 102bd59b0566..000000000000 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqExceptionTranslator.java +++ /dev/null @@ -1,50 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.autoconfigure.jooq; - -import java.sql.SQLException; - -import org.apache.commons.logging.LogFactory; -import org.jooq.ExecuteContext; -import org.jooq.ExecuteListener; - -import org.springframework.dao.DataAccessException; - -/** - * Transforms {@link SQLException} into a Spring-specific {@link DataAccessException}. - * - * @author Lukas Eder - * @author Andreas Ahlenstorf - * @author Phillip Webb - * @author Stephane Nicoll - * @since 1.5.10 - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of - * {@link ExceptionTranslatorExecuteListener#DEFAULT} or - * {@link ExceptionTranslatorExecuteListener#of} - */ -@Deprecated(since = "3.3.0", forRemoval = true) -public class JooqExceptionTranslator implements ExecuteListener { - - private final DefaultExceptionTranslatorExecuteListener delegate = new DefaultExceptionTranslatorExecuteListener( - LogFactory.getLog(JooqExceptionTranslator.class)); - - @Override - public void exception(ExecuteContext context) { - this.delegate.exception(context); - } - -} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqProperties.java index ccf73e8ea2fb..a8179ea21e96 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/JooqProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,15 +21,17 @@ import org.jooq.SQLDialect; import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.core.io.Resource; /** - * Configuration properties for the JOOQ database library. + * Configuration properties for the jOOQ database library. * * @author Andreas Ahlenstorf * @author Michael Simons + * @author Moritz Halbritter * @since 1.3.0 */ -@ConfigurationProperties(prefix = "spring.jooq") +@ConfigurationProperties("spring.jooq") public class JooqProperties { /** @@ -37,6 +39,11 @@ public class JooqProperties { */ private SQLDialect sqlDialect; + /** + * Location of the jOOQ config file. + */ + private Resource config; + public SQLDialect getSqlDialect() { return this.sqlDialect; } @@ -45,6 +52,14 @@ public void setSqlDialect(SQLDialect sqlDialect) { this.sqlDialect = sqlDialect; } + public Resource getConfig() { + return this.config; + } + + public void setConfig(Resource config) { + this.config = config; + } + /** * Determine the {@link SQLDialect} to use based on this configuration and the primary * {@link DataSource}. diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/SpringTransaction.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/SpringTransaction.java index 8ad4e14b4a32..d23fa93e4d27 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/SpringTransaction.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/SpringTransaction.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,7 +21,7 @@ import org.springframework.transaction.TransactionStatus; /** - * Adapts a Spring transaction for JOOQ. + * Adapts a Spring transaction for jOOQ. * * @author Lukas Eder * @author Andreas Ahlenstorf diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/SpringTransactionProvider.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/SpringTransactionProvider.java index f5c51e60c181..6c86f03d2b63 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/SpringTransactionProvider.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/SpringTransactionProvider.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ import org.springframework.transaction.support.DefaultTransactionDefinition; /** - * Allows Spring Transaction to be used with JOOQ. + * Allows Spring Transaction to be used with jOOQ. * * @author Lukas Eder * @author Andreas Ahlenstorf diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/package-info.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/package-info.java index 8f455ab5bbb0..d76bac042208 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/package-info.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/jooq/package-info.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -15,6 +15,6 @@ */ /** - * Auto-configuration for JOOQ. + * Auto-configuration for jOOQ. */ package org.springframework.boot.autoconfigure.jooq; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/ConcurrentKafkaListenerContainerFactoryConfigurer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/ConcurrentKafkaListenerContainerFactoryConfigurer.java index d4c4ebf55d34..ac49dd7af4a6 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/ConcurrentKafkaListenerContainerFactoryConfigurer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/ConcurrentKafkaListenerContainerFactoryConfigurer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -237,6 +237,7 @@ private void configureContainer(ContainerProperties container) { map.from(properties::isMissingTopicsFatal).to(container::setMissingTopicsFatal); map.from(properties::isImmediateStop).to(container::setStopImmediate); map.from(properties::isObservationEnabled).to(container::setObservationEnabled); + map.from(properties::getAuthExceptionRetryInterval).to(container::setAuthExceptionRetryInterval); map.from(this.transactionManager).to(container::setKafkaAwareTransactionManager); map.from(this.rebalanceListener).to(container::setConsumerRebalanceListener); map.from(this.listenerTaskExecutor).to(container::setListenerTaskExecutor); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaAnnotationDrivenConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaAnnotationDrivenConfiguration.java index abbc834f466f..94a746901d35 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaAnnotationDrivenConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaAnnotationDrivenConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,7 +23,6 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnThreading; import org.springframework.boot.autoconfigure.thread.Threading; -import org.springframework.boot.ssl.SslBundles; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.core.task.SimpleAsyncTaskExecutor; @@ -152,11 +151,10 @@ private ConcurrentKafkaListenerContainerFactoryConfigurer configurer() { ConcurrentKafkaListenerContainerFactory kafkaListenerContainerFactory( ConcurrentKafkaListenerContainerFactoryConfigurer configurer, ObjectProvider> kafkaConsumerFactory, - ObjectProvider>> kafkaContainerCustomizer, - ObjectProvider sslBundles) { + ObjectProvider>> kafkaContainerCustomizer) { ConcurrentKafkaListenerContainerFactory factory = new ConcurrentKafkaListenerContainerFactory<>(); - configurer.configure(factory, kafkaConsumerFactory.getIfAvailable(() -> new DefaultKafkaConsumerFactory<>( - this.properties.buildConsumerProperties(sslBundles.getIfAvailable())))); + configurer.configure(factory, kafkaConsumerFactory + .getIfAvailable(() -> new DefaultKafkaConsumerFactory<>(this.properties.buildConsumerProperties()))); kafkaContainerCustomizer.ifAvailable(factory::setContainerCustomizer); return factory; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaAutoConfiguration.java index ba4110407cdf..a8a20baece47 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaAutoConfiguration.java @@ -23,6 +23,7 @@ import org.apache.kafka.clients.CommonClientConfigs; import org.apache.kafka.clients.consumer.ConsumerConfig; import org.apache.kafka.clients.producer.ProducerConfig; +import org.apache.kafka.common.config.SslConfigs; import org.springframework.aot.hint.MemberCategory; import org.springframework.aot.hint.RuntimeHints; @@ -30,14 +31,17 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnSingleCandidate; +import org.springframework.boot.autoconfigure.kafka.KafkaConnectionDetails.Configuration; import org.springframework.boot.autoconfigure.kafka.KafkaProperties.Jaas; import org.springframework.boot.autoconfigure.kafka.KafkaProperties.Retry.Topic.Backoff; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.boot.context.properties.PropertyMapper; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.ssl.SslBundles; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Import; @@ -57,6 +61,7 @@ import org.springframework.kafka.transaction.KafkaTransactionManager; import org.springframework.retry.backoff.BackOffPolicyBuilder; import org.springframework.retry.backoff.SleepingBackOffPolicy; +import org.springframework.util.StringUtils; /** * {@link EnableAutoConfiguration Auto-configuration} for Apache Kafka. @@ -88,8 +93,9 @@ public class KafkaAutoConfiguration { @Bean @ConditionalOnMissingBean(KafkaConnectionDetails.class) - PropertiesKafkaConnectionDetails kafkaConnectionDetails(KafkaProperties properties) { - return new PropertiesKafkaConnectionDetails(properties); + PropertiesKafkaConnectionDetails kafkaConnectionDetails(KafkaProperties properties, + ObjectProvider sslBundles) { + return new PropertiesKafkaConnectionDetails(properties, sslBundles.getIfAvailable()); } @Bean @@ -115,9 +121,9 @@ public LoggingProducerListener kafkaProducerListener() { @Bean @ConditionalOnMissingBean(ConsumerFactory.class) - public DefaultKafkaConsumerFactory kafkaConsumerFactory(KafkaConnectionDetails connectionDetails, - ObjectProvider customizers, ObjectProvider sslBundles) { - Map properties = this.properties.buildConsumerProperties(sslBundles.getIfAvailable()); + DefaultKafkaConsumerFactory kafkaConsumerFactory(KafkaConnectionDetails connectionDetails, + ObjectProvider customizers) { + Map properties = this.properties.buildConsumerProperties(); applyKafkaConnectionDetailsForConsumer(properties, connectionDetails); DefaultKafkaConsumerFactory factory = new DefaultKafkaConsumerFactory<>(properties); customizers.orderedStream().forEach((customizer) -> customizer.customize(factory)); @@ -126,9 +132,9 @@ public LoggingProducerListener kafkaProducerListener() { @Bean @ConditionalOnMissingBean(ProducerFactory.class) - public DefaultKafkaProducerFactory kafkaProducerFactory(KafkaConnectionDetails connectionDetails, - ObjectProvider customizers, ObjectProvider sslBundles) { - Map properties = this.properties.buildProducerProperties(sslBundles.getIfAvailable()); + DefaultKafkaProducerFactory kafkaProducerFactory(KafkaConnectionDetails connectionDetails, + ObjectProvider customizers) { + Map properties = this.properties.buildProducerProperties(); applyKafkaConnectionDetailsForProducer(properties, connectionDetails); DefaultKafkaProducerFactory factory = new DefaultKafkaProducerFactory<>(properties); String transactionIdPrefix = this.properties.getProducer().getTransactionIdPrefix(); @@ -147,7 +153,7 @@ public LoggingProducerListener kafkaProducerListener() { } @Bean - @ConditionalOnProperty(name = "spring.kafka.jaas.enabled") + @ConditionalOnBooleanProperty("spring.kafka.jaas.enabled") @ConditionalOnMissingBean public KafkaJaasLoginModuleInitializer kafkaJaasInitializer() throws IOException { KafkaJaasLoginModuleInitializer jaas = new KafkaJaasLoginModuleInitializer(); @@ -164,8 +170,8 @@ public KafkaJaasLoginModuleInitializer kafkaJaasInitializer() throws IOException @Bean @ConditionalOnMissingBean - public KafkaAdmin kafkaAdmin(KafkaConnectionDetails connectionDetails, ObjectProvider sslBundles) { - Map properties = this.properties.buildAdminProperties(sslBundles.getIfAvailable()); + KafkaAdmin kafkaAdmin(KafkaConnectionDetails connectionDetails) { + Map properties = this.properties.buildAdminProperties(null); applyKafkaConnectionDetailsForAdmin(properties, connectionDetails); KafkaAdmin kafkaAdmin = new KafkaAdmin(properties); KafkaProperties.Admin admin = this.properties.getAdmin(); @@ -182,7 +188,7 @@ public KafkaAdmin kafkaAdmin(KafkaConnectionDetails connectionDetails, ObjectPro } @Bean - @ConditionalOnProperty(name = "spring.kafka.retry.topic.enabled") + @ConditionalOnBooleanProperty("spring.kafka.retry.topic.enabled") @ConditionalOnSingleCandidate(KafkaTemplate.class) public RetryTopicConfiguration kafkaRetryTopicConfiguration(KafkaTemplate kafkaTemplate) { KafkaProperties.Retry.Topic retryTopic = this.properties.getRetry().getTopic(); @@ -197,26 +203,26 @@ public RetryTopicConfiguration kafkaRetryTopicConfiguration(KafkaTemplate private void applyKafkaConnectionDetailsForConsumer(Map properties, KafkaConnectionDetails connectionDetails) { - properties.put(ConsumerConfig.BOOTSTRAP_SERVERS_CONFIG, connectionDetails.getConsumerBootstrapServers()); - if (!(connectionDetails instanceof PropertiesKafkaConnectionDetails)) { - properties.put(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); - } + Configuration consumer = connectionDetails.getConsumer(); + properties.put(ConsumerConfig.BOOTSTRAP_SERVERS_CONFIG, consumer.getBootstrapServers()); + applySecurityProtocol(properties, connectionDetails.getSecurityProtocol()); + applySslBundle(properties, consumer.getSslBundle()); } private void applyKafkaConnectionDetailsForProducer(Map properties, KafkaConnectionDetails connectionDetails) { - properties.put(ProducerConfig.BOOTSTRAP_SERVERS_CONFIG, connectionDetails.getProducerBootstrapServers()); - if (!(connectionDetails instanceof PropertiesKafkaConnectionDetails)) { - properties.put(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); - } + Configuration producer = connectionDetails.getProducer(); + properties.put(ProducerConfig.BOOTSTRAP_SERVERS_CONFIG, producer.getBootstrapServers()); + applySecurityProtocol(properties, producer.getSecurityProtocol()); + applySslBundle(properties, producer.getSslBundle()); } private void applyKafkaConnectionDetailsForAdmin(Map properties, KafkaConnectionDetails connectionDetails) { - properties.put(CommonClientConfigs.BOOTSTRAP_SERVERS_CONFIG, connectionDetails.getAdminBootstrapServers()); - if (!(connectionDetails instanceof PropertiesKafkaConnectionDetails)) { - properties.put(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); - } + Configuration admin = connectionDetails.getAdmin(); + properties.put(CommonClientConfigs.BOOTSTRAP_SERVERS_CONFIG, admin.getBootstrapServers()); + applySecurityProtocol(properties, admin.getSecurityProtocol()); + applySslBundle(properties, admin.getSslBundle()); } private static void setBackOffPolicy(RetryTopicConfigurationBuilder builder, Backoff retryTopicBackoff) { @@ -235,6 +241,19 @@ private static void setBackOffPolicy(RetryTopicConfigurationBuilder builder, Bac } } + static void applySslBundle(Map properties, SslBundle sslBundle) { + if (sslBundle != null) { + properties.put(SslConfigs.SSL_ENGINE_FACTORY_CLASS_CONFIG, SslBundleSslEngineFactory.class); + properties.put(SslBundle.class.getName(), sslBundle); + } + } + + static void applySecurityProtocol(Map properties, String securityProtocol) { + if (StringUtils.hasLength(securityProtocol)) { + properties.put(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, securityProtocol); + } + } + static class KafkaRuntimeHints implements RuntimeHintsRegistrar { @Override diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaConnectionDetails.java index 5490e9852e8f..5f274f891a3a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,7 @@ import java.util.List; import org.springframework.boot.autoconfigure.service.connection.ConnectionDetails; +import org.springframework.boot.ssl.SslBundle; /** * Details required to establish a connection to a Kafka service. @@ -36,36 +37,173 @@ public interface KafkaConnectionDetails extends ConnectionDetails { */ List getBootstrapServers(); + /** + * Returns the SSL bundle. + * @return the SSL bundle + * @since 3.5.0 + */ + default SslBundle getSslBundle() { + return null; + } + + /** + * Returns the security protocol. + * @return the security protocol + * @since 3.5.0 + */ + default String getSecurityProtocol() { + return null; + } + + /** + * Returns the consumer configuration. + * @return the consumer configuration + * @since 3.5.0 + */ + default Configuration getConsumer() { + return Configuration.of(getBootstrapServers(), getSslBundle(), getSecurityProtocol()); + } + + /** + * Returns the producer configuration. + * @return the producer configuration + * @since 3.5.0 + */ + default Configuration getProducer() { + return Configuration.of(getBootstrapServers(), getSslBundle(), getSecurityProtocol()); + } + + /** + * Returns the admin configuration. + * @return the admin configuration + * @since 3.5.0 + */ + default Configuration getAdmin() { + return Configuration.of(getBootstrapServers(), getSslBundle(), getSecurityProtocol()); + } + + /** + * Returns the Kafka Streams configuration. + * @return the Kafka Streams configuration + * @since 3.5.0 + */ + default Configuration getStreams() { + return Configuration.of(getBootstrapServers(), getSslBundle(), getSecurityProtocol()); + } + /** * Returns the list of bootstrap servers used for consumers. * @return the list of bootstrap servers used for consumers + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of {@link #getConsumer()} */ + @Deprecated(since = "3.5.0", forRemoval = true) default List getConsumerBootstrapServers() { - return getBootstrapServers(); + return getConsumer().getBootstrapServers(); } /** * Returns the list of bootstrap servers used for producers. * @return the list of bootstrap servers used for producers + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of {@link #getProducer()} */ + @Deprecated(since = "3.5.0", forRemoval = true) default List getProducerBootstrapServers() { - return getBootstrapServers(); + return getProducer().getBootstrapServers(); } /** * Returns the list of bootstrap servers used for the admin. * @return the list of bootstrap servers used for the admin + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of {@link #getAdmin()} */ + @Deprecated(since = "3.5.0", forRemoval = true) default List getAdminBootstrapServers() { - return getBootstrapServers(); + return getAdmin().getBootstrapServers(); } /** * Returns the list of bootstrap servers used for Kafka Streams. * @return the list of bootstrap servers used for Kafka Streams + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of {@link #getStreams()} */ + @Deprecated(since = "3.5.0", forRemoval = true) default List getStreamsBootstrapServers() { - return getBootstrapServers(); + return getStreams().getBootstrapServers(); + } + + /** + * Kafka connection details configuration. + */ + interface Configuration { + + /** + * Creates a new configuration with the given bootstrap servers. + * @param bootstrapServers the bootstrap servers + * @return the configuration + */ + static Configuration of(List bootstrapServers) { + return Configuration.of(bootstrapServers, null, null); + } + + /** + * Creates a new configuration with the given bootstrap servers and SSL bundle. + * @param bootstrapServers the bootstrap servers + * @param sslBundle the SSL bundle + * @return the configuration + */ + static Configuration of(List bootstrapServers, SslBundle sslBundle) { + return Configuration.of(bootstrapServers, sslBundle, null); + } + + /** + * Creates a new configuration with the given bootstrap servers, SSL bundle and + * security protocol. + * @param bootstrapServers the bootstrap servers + * @param sslBundle the SSL bundle + * @param securityProtocol the security protocol + * @return the configuration + */ + static Configuration of(List bootstrapServers, SslBundle sslBundle, String securityProtocol) { + return new Configuration() { + @Override + public List getBootstrapServers() { + return bootstrapServers; + } + + @Override + public SslBundle getSslBundle() { + return sslBundle; + } + + @Override + public String getSecurityProtocol() { + return securityProtocol; + } + }; + } + + /** + * Returns the list of bootstrap servers. + * @return the list of bootstrap servers + */ + List getBootstrapServers(); + + /** + * Returns the SSL bundle. + * @return the SSL bundle + */ + default SslBundle getSslBundle() { + return null; + } + + /** + * Returns the security protocol. + * @return the security protocol + */ + default String getSecurityProtocol() { + return null; + } + } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaProperties.java index 61e06b015119..95113f0dbf95 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaProperties.java @@ -38,7 +38,6 @@ import org.springframework.boot.context.properties.PropertyMapper; import org.springframework.boot.context.properties.source.MutuallyExclusiveConfigurationPropertiesException; import org.springframework.boot.convert.DurationUnit; -import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.ssl.SslBundles; import org.springframework.core.io.Resource; import org.springframework.kafka.listener.ContainerProperties.AckMode; @@ -60,9 +59,10 @@ * @author Tomaz Fernandes * @author Andy Wilkinson * @author Scott Frederick + * @author Yanming Zhou * @since 1.5.0 */ -@ConfigurationProperties(prefix = "spring.kafka") +@ConfigurationProperties("spring.kafka") public class KafkaProperties { /** @@ -338,6 +338,12 @@ public static class Consumer { */ private Integer maxPollRecords; + /** + * Maximum delay between invocations of poll() when using consumer group + * management. + */ + private Duration maxPollInterval; + /** * Additional consumer-specific properties used to configure the client. */ @@ -455,6 +461,14 @@ public void setMaxPollRecords(Integer maxPollRecords) { this.maxPollRecords = maxPollRecords; } + public Duration getMaxPollInterval() { + return this.maxPollInterval; + } + + public void setMaxPollInterval(Duration maxPollInterval) { + this.maxPollInterval = maxPollInterval; + } + public Map getProperties() { return this.properties; } @@ -484,6 +498,9 @@ public Map buildProperties(SslBundles sslBundles) { map.from(this::getKeyDeserializer).to(properties.in(ConsumerConfig.KEY_DESERIALIZER_CLASS_CONFIG)); map.from(this::getValueDeserializer).to(properties.in(ConsumerConfig.VALUE_DESERIALIZER_CLASS_CONFIG)); map.from(this::getMaxPollRecords).to(properties.in(ConsumerConfig.MAX_POLL_RECORDS_CONFIG)); + map.from(this::getMaxPollInterval) + .asInt(Duration::toMillis) + .to(properties.in(ConsumerConfig.MAX_POLL_INTERVAL_MS_CONFIG)); return properties.with(this.ssl, this.security, this.properties, sslBundles); } @@ -1081,6 +1098,11 @@ public enum Type { */ private boolean observationEnabled; + /** + * Time between retries after authentication exceptions. + */ + private Duration authExceptionRetryInterval; + public Type getType() { return this.type; } @@ -1233,6 +1255,14 @@ public void setObservationEnabled(boolean observationEnabled) { this.observationEnabled = observationEnabled; } + public Duration getAuthExceptionRetryInterval() { + return this.authExceptionRetryInterval; + } + + public void setAuthExceptionRetryInterval(Duration authExceptionRetryInterval) { + this.authExceptionRetryInterval = authExceptionRetryInterval; + } + } public static class Ssl { @@ -1401,10 +1431,10 @@ public Map buildProperties() { public Map buildProperties(SslBundles sslBundles) { validate(); String bundleName = getBundle(); + Properties properties = new Properties(); if (StringUtils.hasText(bundleName)) { - return buildPropertiesForSslBundle(sslBundles, bundleName); + return properties; } - Properties properties = new Properties(); PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull(); map.from(this::getKeyPassword).to(properties.in(SslConfigs.SSL_KEY_PASSWORD_CONFIG)); map.from(this::getKeyStoreCertificateChain) @@ -1425,13 +1455,6 @@ public Map buildProperties(SslBundles sslBundles) { return properties; } - private Map buildPropertiesForSslBundle(SslBundles sslBundles, String name) { - Properties properties = new Properties(); - properties.in(SslConfigs.SSL_ENGINE_FACTORY_CLASS_CONFIG).accept(SslBundleSslEngineFactory.class); - properties.in(SslBundle.class.getName()).accept(sslBundles.getBundle(name)); - return properties; - } - private void validate() { MutuallyExclusiveConfigurationPropertiesException.throwIfMultipleMatchingValuesIn((entries) -> { entries.put("spring.kafka.ssl.key-store-key", getKeyStoreKey()); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaStreamsAnnotationDrivenConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaStreamsAnnotationDrivenConfiguration.java index 701384326303..a3078a0dad36 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaStreamsAnnotationDrivenConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/KafkaStreamsAnnotationDrivenConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,7 +18,6 @@ import java.util.Map; -import org.apache.kafka.clients.CommonClientConfigs; import org.apache.kafka.clients.consumer.ConsumerConfig; import org.apache.kafka.streams.StreamsBuilder; import org.apache.kafka.streams.StreamsConfig; @@ -30,7 +29,6 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.context.properties.source.InvalidConfigurationPropertyValueException; -import org.springframework.boot.ssl.SslBundles; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.core.env.Environment; @@ -63,8 +61,8 @@ class KafkaStreamsAnnotationDrivenConfiguration { @ConditionalOnMissingBean @Bean(KafkaStreamsDefaultConfiguration.DEFAULT_STREAMS_CONFIG_BEAN_NAME) KafkaStreamsConfiguration defaultKafkaStreamsConfig(Environment environment, - KafkaConnectionDetails connectionDetails, ObjectProvider sslBundles) { - Map properties = this.properties.buildStreamsProperties(sslBundles.getIfAvailable()); + KafkaConnectionDetails connectionDetails) { + Map properties = this.properties.buildStreamsProperties(null); applyKafkaConnectionDetailsForStreams(properties, connectionDetails); if (this.properties.getStreams().getApplicationId() == null) { String applicationName = environment.getProperty("spring.application.name"); @@ -87,10 +85,10 @@ KafkaStreamsFactoryBeanConfigurer kafkaStreamsFactoryBeanConfigurer( private void applyKafkaConnectionDetailsForStreams(Map properties, KafkaConnectionDetails connectionDetails) { - properties.put(ConsumerConfig.BOOTSTRAP_SERVERS_CONFIG, connectionDetails.getStreamsBootstrapServers()); - if (!(connectionDetails instanceof PropertiesKafkaConnectionDetails)) { - properties.put(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); - } + KafkaConnectionDetails.Configuration streams = connectionDetails.getStreams(); + properties.put(ConsumerConfig.BOOTSTRAP_SERVERS_CONFIG, streams.getBootstrapServers()); + KafkaAutoConfiguration.applySecurityProtocol(properties, streams.getSecurityProtocol()); + KafkaAutoConfiguration.applySslBundle(properties, streams.getSslBundle()); } // Separate class required to avoid BeanCurrentlyInCreationException diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/PropertiesKafkaConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/PropertiesKafkaConnectionDetails.java index 26ea6ad67f5c..310e107aafb0 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/PropertiesKafkaConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/kafka/PropertiesKafkaConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,6 +18,12 @@ import java.util.List; +import org.springframework.boot.autoconfigure.kafka.KafkaProperties.Ssl; +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslBundles; +import org.springframework.util.Assert; +import org.springframework.util.StringUtils; + /** * Adapts {@link KafkaProperties} to {@link KafkaConnectionDetails}. * @@ -29,8 +35,11 @@ class PropertiesKafkaConnectionDetails implements KafkaConnectionDetails { private final KafkaProperties properties; - PropertiesKafkaConnectionDetails(KafkaProperties properties) { + private final SslBundles sslBundles; + + PropertiesKafkaConnectionDetails(KafkaProperties properties, SslBundles sslBundles) { this.properties = properties; + this.sslBundles = sslBundles; } @Override @@ -39,22 +48,59 @@ public List getBootstrapServers() { } @Override - public List getConsumerBootstrapServers() { - return getServers(this.properties.getConsumer().getBootstrapServers()); + public Configuration getConsumer() { + List servers = this.properties.getConsumer().getBootstrapServers(); + SslBundle sslBundle = getBundle(this.properties.getConsumer().getSsl()); + String protocol = this.properties.getConsumer().getSecurity().getProtocol(); + return Configuration.of((servers != null) ? servers : getBootstrapServers(), + (sslBundle != null) ? sslBundle : getSslBundle(), + (StringUtils.hasLength(protocol)) ? protocol : getSecurityProtocol()); + } + + @Override + public Configuration getProducer() { + List servers = this.properties.getProducer().getBootstrapServers(); + SslBundle sslBundle = getBundle(this.properties.getProducer().getSsl()); + String protocol = this.properties.getProducer().getSecurity().getProtocol(); + return Configuration.of((servers != null) ? servers : getBootstrapServers(), + (sslBundle != null) ? sslBundle : getSslBundle(), + (StringUtils.hasLength(protocol)) ? protocol : getSecurityProtocol()); + } + + @Override + public Configuration getStreams() { + List servers = this.properties.getStreams().getBootstrapServers(); + SslBundle sslBundle = getBundle(this.properties.getStreams().getSsl()); + String protocol = this.properties.getStreams().getSecurity().getProtocol(); + return Configuration.of((servers != null) ? servers : getBootstrapServers(), + (sslBundle != null) ? sslBundle : getSslBundle(), + (StringUtils.hasLength(protocol)) ? protocol : getSecurityProtocol()); + } + + @Override + public Configuration getAdmin() { + SslBundle sslBundle = getBundle(this.properties.getAdmin().getSsl()); + String protocol = this.properties.getAdmin().getSecurity().getProtocol(); + return Configuration.of(getBootstrapServers(), (sslBundle != null) ? sslBundle : getSslBundle(), + (StringUtils.hasLength(protocol)) ? protocol : getSecurityProtocol()); } @Override - public List getProducerBootstrapServers() { - return getServers(this.properties.getProducer().getBootstrapServers()); + public SslBundle getSslBundle() { + return getBundle(this.properties.getSsl()); } @Override - public List getStreamsBootstrapServers() { - return getServers(this.properties.getStreams().getBootstrapServers()); + public String getSecurityProtocol() { + return this.properties.getSecurity().getProtocol(); } - private List getServers(List servers) { - return (servers != null) ? servers : getBootstrapServers(); + private SslBundle getBundle(Ssl ssl) { + if (StringUtils.hasLength(ssl.getBundle())) { + Assert.notNull(this.sslBundles, "SSL bundle name has been set but no SSL bundles found in context"); + return this.sslBundles.getBundle(ssl.getBundle()); + } + return null; } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/LdapAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/LdapAutoConfiguration.java index 205af4c3e0ac..07d43eb9d35d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/LdapAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/LdapAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,6 +17,7 @@ package org.springframework.boot.autoconfigure.ldap; import java.util.Collections; +import java.util.Locale; import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; @@ -26,13 +27,17 @@ import org.springframework.boot.autoconfigure.ldap.LdapProperties.Template; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.boot.context.properties.PropertyMapper; +import org.springframework.boot.convert.ApplicationConversionService; import org.springframework.context.annotation.Bean; import org.springframework.core.env.Environment; +import org.springframework.ldap.convert.ConverterUtils; import org.springframework.ldap.core.ContextSource; import org.springframework.ldap.core.LdapOperations; import org.springframework.ldap.core.LdapTemplate; import org.springframework.ldap.core.support.DirContextAuthenticationStrategy; import org.springframework.ldap.core.support.LdapContextSource; +import org.springframework.ldap.odm.core.ObjectDirectoryMapper; +import org.springframework.ldap.odm.core.impl.DefaultObjectDirectoryMapper; /** * {@link EnableAutoConfiguration Auto-configuration} for LDAP. @@ -63,6 +68,9 @@ public LdapContextSource ldapContextSource(LdapConnectionDetails connectionDetai propertyMapper.from(connectionDetails.getUsername()).to(source::setUserDn); propertyMapper.from(connectionDetails.getPassword()).to(source::setPassword); propertyMapper.from(properties.getAnonymousReadOnly()).to(source::setAnonymousReadOnly); + propertyMapper.from(properties.getReferral()) + .as(((referral) -> referral.name().toLowerCase(Locale.ROOT))) + .to(source::setReferral); propertyMapper.from(connectionDetails.getBase()).to(source::setBase); propertyMapper.from(connectionDetails.getUrls()).to(source::setUrls); propertyMapper.from(properties.getBaseEnvironment()) @@ -70,12 +78,24 @@ public LdapContextSource ldapContextSource(LdapConnectionDetails connectionDetai return source; } + @Bean + @ConditionalOnMissingBean + public ObjectDirectoryMapper objectDirectoryMapper() { + ApplicationConversionService conversionService = new ApplicationConversionService(); + ConverterUtils.addDefaultConverters(conversionService); + DefaultObjectDirectoryMapper objectDirectoryMapper = new DefaultObjectDirectoryMapper(); + objectDirectoryMapper.setConversionService(conversionService); + return objectDirectoryMapper; + } + @Bean @ConditionalOnMissingBean(LdapOperations.class) - public LdapTemplate ldapTemplate(LdapProperties properties, ContextSource contextSource) { + public LdapTemplate ldapTemplate(LdapProperties properties, ContextSource contextSource, + ObjectDirectoryMapper objectDirectoryMapper) { Template template = properties.getTemplate(); PropertyMapper propertyMapper = PropertyMapper.get().alwaysApplyingWhenNonNull(); LdapTemplate ldapTemplate = new LdapTemplate(contextSource); + ldapTemplate.setObjectDirectoryMapper(objectDirectoryMapper); propertyMapper.from(template.isIgnorePartialResultException()) .to(ldapTemplate::setIgnorePartialResultException); propertyMapper.from(template.isIgnoreNameNotFoundException()).to(ldapTemplate::setIgnoreNameNotFoundException); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/LdapProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/LdapProperties.java index ec59ac43bc2c..8ca76d59abcd 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/LdapProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/LdapProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,6 +21,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.core.env.Environment; +import org.springframework.ldap.ReferralException; import org.springframework.ldap.core.LdapTemplate; import org.springframework.util.Assert; import org.springframework.util.ObjectUtils; @@ -31,7 +32,7 @@ * @author Eddú Meléndez * @since 1.5.0 */ -@ConfigurationProperties(prefix = "spring.ldap") +@ConfigurationProperties("spring.ldap") public class LdapProperties { private static final int DEFAULT_PORT = 389; @@ -62,6 +63,12 @@ public class LdapProperties { */ private Boolean anonymousReadOnly; + /** + * Specify how referrals encountered by the service provider are to be processed. If + * not specified, the default is determined by the provider. + */ + private Referral referral; + /** * LDAP specification settings. */ @@ -109,6 +116,14 @@ public void setAnonymousReadOnly(Boolean anonymousReadOnly) { this.anonymousReadOnly = anonymousReadOnly; } + public Referral getReferral() { + return this.referral; + } + + public void setReferral(Referral referral) { + this.referral = referral; + } + public Map getBaseEnvironment() { return this.baseEnvironment; } @@ -125,7 +140,7 @@ public String[] determineUrls(Environment environment) { } private int determinePort(Environment environment) { - Assert.notNull(environment, "Environment must not be null"); + Assert.notNull(environment, "'environment' must not be null"); String localPort = environment.getProperty("local.ldap.port"); if (localPort != null) { return Integer.parseInt(localPort); @@ -182,4 +197,26 @@ public void setIgnoreSizeLimitExceededException(Boolean ignoreSizeLimitExceededE } + /** + * Define the methods to handle referrals. + */ + public enum Referral { + + /** + * Follow referrals automatically. + */ + FOLLOW, + + /** + * Ignore referrals. + */ + IGNORE, + + /** + * Throw {@link ReferralException} when a referral is encountered. + */ + THROW + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/embedded/EmbeddedLdapAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/embedded/EmbeddedLdapAutoConfiguration.java index 384d4ba0e955..7a36b6f67985 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/embedded/EmbeddedLdapAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/embedded/EmbeddedLdapAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,10 +28,10 @@ import com.unboundid.ldap.sdk.LDAPException; import com.unboundid.ldap.sdk.schema.Schema; import com.unboundid.ldif.LDIFReader; -import jakarta.annotation.PreDestroy; import org.springframework.aot.hint.RuntimeHints; import org.springframework.aot.hint.RuntimeHintsRegistrar; +import org.springframework.beans.factory.DisposableBean; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionMessage; @@ -77,7 +77,7 @@ @ConditionalOnClass(InMemoryDirectoryServer.class) @Conditional(EmbeddedLdapAutoConfiguration.EmbeddedLdapCondition.class) @ImportRuntimeHints(EmbeddedLdapAutoConfigurationRuntimeHints.class) -public class EmbeddedLdapAutoConfiguration { +public class EmbeddedLdapAutoConfiguration implements DisposableBean { private static final String PROPERTY_SOURCE_NAME = "ldap.ports"; @@ -167,8 +167,8 @@ private Map getLdapPorts(MutablePropertySources sources) { return (Map) propertySource.getSource(); } - @PreDestroy - public void close() { + @Override + public void destroy() throws Exception { if (this.server != null) { this.server.shutDown(true); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/embedded/EmbeddedLdapProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/embedded/EmbeddedLdapProperties.java index 7018b92d7393..beb25e69e125 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/embedded/EmbeddedLdapProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ldap/embedded/EmbeddedLdapProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ * @author Mathieu Ouellet * @since 1.5.0 */ -@ConfigurationProperties(prefix = "spring.ldap.embedded") +@ConfigurationProperties("spring.ldap.embedded") public class EmbeddedLdapProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseAutoConfiguration.java index 469f6a93e94b..4367e65e2386 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,6 +33,7 @@ import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.AnyNestedCondition; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; @@ -74,7 +75,7 @@ */ @AutoConfiguration(after = { DataSourceAutoConfiguration.class, HibernateJpaAutoConfiguration.class }) @ConditionalOnClass({ SpringLiquibase.class, DatabaseChange.class }) -@ConditionalOnProperty(prefix = "spring.liquibase", name = "enabled", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.liquibase.enabled", matchIfMissing = true) @Conditional(LiquibaseDataSourceCondition.class) @Import(DatabaseInitializationDependencyConfigurer.class) @ImportRuntimeHints(LiquibaseAutoConfigurationRuntimeHints.class) @@ -133,6 +134,12 @@ SpringLiquibase liquibase(ObjectProvider dataSource, if (properties.getUiService() != null) { liquibase.setUiService(UIServiceEnum.valueOf(properties.getUiService().name())); } + if (properties.getAnalyticsEnabled() != null) { + liquibase.setAnalyticsEnabled(properties.getAnalyticsEnabled()); + } + if (properties.getLicenseKey() != null) { + liquibase.setLicenseKey(properties.getLicenseKey()); + } customizers.orderedStream().forEach((customizer) -> customizer.customize(liquibase)); return liquibase; } @@ -209,7 +216,7 @@ private static final class JdbcConnectionDetailsCondition { } - @ConditionalOnProperty(prefix = "spring.liquibase", name = "url") + @ConditionalOnProperty("spring.liquibase.url") private static final class LiquibaseUrlCondition { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseProperties.java index 7d1df833225a..c9571a70cf3f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -154,12 +154,22 @@ public class LiquibaseProperties { */ private UiService uiService; + /** + * Whether to send product usage data and analytics to Liquibase. + */ + private Boolean analyticsEnabled; + + /** + * Liquibase Pro license key. + */ + private String licenseKey; + public String getChangeLog() { return this.changeLog; } public void setChangeLog(String changeLog) { - Assert.notNull(changeLog, "ChangeLog must not be null"); + Assert.notNull(changeLog, "'changeLog' must not be null"); this.changeLog = changeLog; } @@ -331,6 +341,22 @@ public void setUiService(UiService uiService) { this.uiService = uiService; } + public Boolean getAnalyticsEnabled() { + return this.analyticsEnabled; + } + + public void setAnalyticsEnabled(Boolean analyticsEnabled) { + this.analyticsEnabled = analyticsEnabled; + } + + public String getLicenseKey() { + return this.licenseKey; + } + + public void setLicenseKey(String licenseKey) { + this.licenseKey = licenseKey; + } + /** * Enumeration of types of summary to show. Values are the same as those on * {@link UpdateSummaryEnum}. To maximize backwards compatibility, the Liquibase enum diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLogger.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLogger.java index 1af820991ab5..4cd8e566ce03 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLogger.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLogger.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,13 +43,13 @@ class ConditionEvaluationReportLogger { private final LogLevel logLevel; ConditionEvaluationReportLogger(LogLevel logLevel, Supplier reportSupplier) { - Assert.isTrue(isInfoOrDebug(logLevel), "LogLevel must be INFO or DEBUG"); + Assert.isTrue(isInfoOrDebug(logLevel), "'logLevel' must be INFO or DEBUG"); this.logLevel = logLevel; this.reportSupplier = reportSupplier; } - private boolean isInfoOrDebug(LogLevel logLevelForReport) { - return LogLevel.INFO.equals(logLevelForReport) || LogLevel.DEBUG.equals(logLevelForReport); + private boolean isInfoOrDebug(LogLevel logLevel) { + return LogLevel.INFO.equals(logLevel) || LogLevel.DEBUG.equals(logLevel); } void logReport(boolean isCrashReport) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLoggingListener.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLoggingListener.java index 31bcac1d4f53..d2de269fbe99 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLoggingListener.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLoggingListener.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -50,15 +50,15 @@ public class ConditionEvaluationReportLoggingListener implements ApplicationContextInitializer { - private final LogLevel logLevelForReport; + private final LogLevel logLevel; public ConditionEvaluationReportLoggingListener() { this(LogLevel.DEBUG); } - private ConditionEvaluationReportLoggingListener(LogLevel logLevelForReport) { - Assert.isTrue(isInfoOrDebug(logLevelForReport), "LogLevel must be INFO or DEBUG"); - this.logLevelForReport = logLevelForReport; + private ConditionEvaluationReportLoggingListener(LogLevel logLevel) { + Assert.isTrue(isInfoOrDebug(logLevel), "'logLevel' must be INFO or DEBUG"); + this.logLevel = logLevel; } private boolean isInfoOrDebug(LogLevel logLevelForReport) { @@ -100,8 +100,8 @@ private ConditionEvaluationReportListener(ConfigurableApplicationContext context else { reportSupplier = this::getReport; } - this.logger = new ConditionEvaluationReportLogger( - ConditionEvaluationReportLoggingListener.this.logLevelForReport, reportSupplier); + this.logger = new ConditionEvaluationReportLogger(ConditionEvaluationReportLoggingListener.this.logLevel, + reportSupplier); } private ConditionEvaluationReport getReport() { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailProperties.java index 2b2e182998a2..0db1934e3aa5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ * @author Eddú Meléndez * @since 1.2.0 */ -@ConfigurationProperties(prefix = "spring.mail") +@ConfigurationProperties("spring.mail") public class MailProperties { private static final Charset DEFAULT_CHARSET = StandardCharsets.UTF_8; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderAutoConfiguration.java index 3feb692da60b..6ba7555870cc 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -57,12 +57,12 @@ static class MailSenderCondition extends AnyNestedCondition { super(ConfigurationPhase.PARSE_CONFIGURATION); } - @ConditionalOnProperty(prefix = "spring.mail", name = "host") + @ConditionalOnProperty("spring.mail.host") static class HostProperty { } - @ConditionalOnProperty(prefix = "spring.mail", name = "jndi-name") + @ConditionalOnProperty("spring.mail.jndi-name") static class JndiNameProperty { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderJndiConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderJndiConfiguration.java index 02cfe435caf7..6c9cf2535ebf 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderJndiConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderJndiConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ */ @Configuration(proxyBeanMethods = false) @ConditionalOnClass(Session.class) -@ConditionalOnProperty(prefix = "spring.mail", name = "jndi-name") +@ConditionalOnProperty("spring.mail.jndi-name") @ConditionalOnJndi class MailSenderJndiConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderPropertiesConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderPropertiesConfiguration.java index 94cabfaa1cb1..1e0741decdb1 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderPropertiesConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderPropertiesConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,7 +40,7 @@ * @author Stephane Nicoll */ @Configuration(proxyBeanMethods = false) -@ConditionalOnProperty(prefix = "spring.mail", name = "host") +@ConditionalOnProperty("spring.mail.host") class MailSenderPropertiesConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderValidatorAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderValidatorAutoConfiguration.java index 66fc85674ec7..722eb19571dc 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderValidatorAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mail/MailSenderValidatorAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,7 +20,7 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnSingleCandidate; import org.springframework.mail.javamail.JavaMailSenderImpl; @@ -33,7 +33,7 @@ * @since 1.3.0 */ @AutoConfiguration(after = MailSenderAutoConfiguration.class) -@ConditionalOnProperty(prefix = "spring.mail", value = "test-connection") +@ConditionalOnBooleanProperty("spring.mail.test-connection") @ConditionalOnSingleCandidate(JavaMailSenderImpl.class) public class MailSenderValidatorAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoAutoConfiguration.java index 5905b2f22465..d510c77dd895 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,8 +48,9 @@ public class MongoAutoConfiguration { @Bean @ConditionalOnMissingBean(MongoConnectionDetails.class) - PropertiesMongoConnectionDetails mongoConnectionDetails(MongoProperties properties) { - return new PropertiesMongoConnectionDetails(properties); + PropertiesMongoConnectionDetails mongoConnectionDetails(MongoProperties properties, + ObjectProvider sslBundles) { + return new PropertiesMongoConnectionDetails(properties, sslBundles.getIfAvailable()); } @Bean @@ -70,9 +71,9 @@ MongoClientSettings mongoClientSettings() { @Bean StandardMongoClientSettingsBuilderCustomizer standardMongoSettingsCustomizer(MongoProperties properties, - MongoConnectionDetails connectionDetails, ObjectProvider sslBundles) { - return new StandardMongoClientSettingsBuilderCustomizer(connectionDetails.getConnectionString(), - properties.getUuidRepresentation(), properties.getSsl(), sslBundles.getIfAvailable()); + MongoConnectionDetails connectionDetails) { + return new StandardMongoClientSettingsBuilderCustomizer(connectionDetails, + properties.getUuidRepresentation()); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoConnectionDetails.java index bfd107505a62..c625cc2a4b3a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,7 @@ import com.mongodb.ConnectionString; import org.springframework.boot.autoconfigure.service.connection.ConnectionDetails; +import org.springframework.boot.ssl.SslBundle; /** * Details required to establish a connection to a MongoDB service. @@ -36,6 +37,15 @@ public interface MongoConnectionDetails extends ConnectionDetails { */ ConnectionString getConnectionString(); + /** + * SSL bundle to use. + * @return the SSL bundle to use + * @since 3.5.0 + */ + default SslBundle getSslBundle() { + return null; + } + /** * GridFS configuration. * @return the GridFS configuration or {@code null} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoProperties.java index 740c97dff7f9..207db93377c6 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoProperties.java @@ -38,7 +38,7 @@ * @author Safeer Ansari * @since 1.0.0 */ -@ConfigurationProperties(prefix = "spring.data.mongodb") +@ConfigurationProperties("spring.data.mongodb") public class MongoProperties { /** @@ -51,6 +51,11 @@ public class MongoProperties { */ public static final String DEFAULT_URI = "mongodb://localhost/test"; + /** + * Protocol to be used for the MongoDB connection. Ignored if 'uri' is set. + */ + private String protocol = "mongodb"; + /** * Mongo server host. Ignored if 'uri' is set. */ @@ -117,6 +122,14 @@ public class MongoProperties { */ private Boolean autoIndexCreation; + public void setProtocol(String protocol) { + this.protocol = protocol; + } + + public String getProtocol() { + return this.protocol; + } + public String getHost() { return this.host; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoReactiveAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoReactiveAutoConfiguration.java index 746bbe4587ff..2c3864907dd1 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoReactiveAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/MongoReactiveAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,8 +53,9 @@ public class MongoReactiveAutoConfiguration { @Bean @ConditionalOnMissingBean(MongoConnectionDetails.class) - PropertiesMongoConnectionDetails mongoConnectionDetails(MongoProperties properties) { - return new PropertiesMongoConnectionDetails(properties); + PropertiesMongoConnectionDetails mongoConnectionDetails(MongoProperties properties, + ObjectProvider sslBundles) { + return new PropertiesMongoConnectionDetails(properties, sslBundles.getIfAvailable()); } @Bean @@ -77,9 +78,9 @@ MongoClientSettings mongoClientSettings() { @Bean StandardMongoClientSettingsBuilderCustomizer standardMongoSettingsCustomizer(MongoProperties properties, - MongoConnectionDetails connectionDetails, ObjectProvider sslBundles) { - return new StandardMongoClientSettingsBuilderCustomizer(connectionDetails.getConnectionString(), - properties.getUuidRepresentation(), properties.getSsl(), sslBundles.getIfAvailable()); + MongoConnectionDetails connectionDetails) { + return new StandardMongoClientSettingsBuilderCustomizer(connectionDetails, + properties.getUuidRepresentation()); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/PropertiesMongoConnectionDetails.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/PropertiesMongoConnectionDetails.java index 04436c717b54..a74ccf80f5a6 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/PropertiesMongoConnectionDetails.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/PropertiesMongoConnectionDetails.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,6 +23,10 @@ import com.mongodb.ConnectionString; +import org.springframework.boot.autoconfigure.mongo.MongoProperties.Ssl; +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslBundles; +import org.springframework.util.Assert; import org.springframework.util.StringUtils; /** @@ -38,17 +42,20 @@ public class PropertiesMongoConnectionDetails implements MongoConnectionDetails private final MongoProperties properties; - public PropertiesMongoConnectionDetails(MongoProperties properties) { + private final SslBundles sslBundles; + + public PropertiesMongoConnectionDetails(MongoProperties properties, SslBundles sslBundles) { this.properties = properties; + this.sslBundles = sslBundles; } @Override public ConnectionString getConnectionString() { - // mongodb://[username:password@]host1[:port1][,host2[:port2],...[,hostN[:portN]]][/[database.collection][?options]] + // protocol://[username:password@]host1[:port1][,host2[:port2],...[,hostN[:portN]]][/[database.collection][?options]] if (this.properties.getUri() != null) { return new ConnectionString(this.properties.getUri()); } - StringBuilder builder = new StringBuilder("mongodb://"); + StringBuilder builder = new StringBuilder(getProtocol()).append("://"); if (this.properties.getUsername() != null) { builder.append(encode(this.properties.getUsername())); builder.append(":"); @@ -76,6 +83,14 @@ public ConnectionString getConnectionString() { return new ConnectionString(builder.toString()); } + private String getProtocol() { + String protocol = this.properties.getProtocol(); + if (StringUtils.hasText(protocol)) { + return protocol; + } + return "mongodb"; + } + private String encode(String input) { return URLEncoder.encode(input, StandardCharsets.UTF_8); } @@ -90,6 +105,19 @@ public GridFs getGridFs() { PropertiesMongoConnectionDetails.this.properties.getGridfs().getBucket()); } + @Override + public SslBundle getSslBundle() { + Ssl ssl = this.properties.getSsl(); + if (!ssl.isEnabled()) { + return null; + } + if (StringUtils.hasLength(ssl.getBundle())) { + Assert.notNull(this.sslBundles, "SSL bundle name has been set but no SSL bundles found in context"); + return this.sslBundles.getBundle(ssl.getBundle()); + } + return SslBundle.systemDefault(); + } + private List getOptions() { List options = new ArrayList<>(); if (StringUtils.hasText(this.properties.getReplicaSetName())) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/StandardMongoClientSettingsBuilderCustomizer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/StandardMongoClientSettingsBuilderCustomizer.java index 12e604491d32..34ac7bece4c6 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/StandardMongoClientSettingsBuilderCustomizer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mongo/StandardMongoClientSettingsBuilderCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,26 +41,55 @@ public class StandardMongoClientSettingsBuilderCustomizer implements MongoClient private final UuidRepresentation uuidRepresentation; + private final MongoConnectionDetails connectionDetails; + private final MongoProperties.Ssl ssl; private final SslBundles sslBundles; private int order = 0; + /** + * Create a new instance. + * @param connectionString the connection string + * @param uuidRepresentation the uuid representation + * @param ssl the ssl properties + * @param sslBundles the ssl bundles + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of + * {@link #StandardMongoClientSettingsBuilderCustomizer(MongoConnectionDetails, UuidRepresentation)} + */ + @Deprecated(forRemoval = true, since = "3.5.0") public StandardMongoClientSettingsBuilderCustomizer(ConnectionString connectionString, UuidRepresentation uuidRepresentation, MongoProperties.Ssl ssl, SslBundles sslBundles) { + this.connectionDetails = null; this.connectionString = connectionString; this.uuidRepresentation = uuidRepresentation; this.ssl = ssl; this.sslBundles = sslBundles; } + public StandardMongoClientSettingsBuilderCustomizer(MongoConnectionDetails connectionDetails, + UuidRepresentation uuidRepresentation) { + this.connectionString = null; + this.ssl = null; + this.sslBundles = null; + this.connectionDetails = connectionDetails; + this.uuidRepresentation = uuidRepresentation; + } + @Override public void customize(MongoClientSettings.Builder settingsBuilder) { settingsBuilder.uuidRepresentation(this.uuidRepresentation); - settingsBuilder.applyConnectionString(this.connectionString); - if (this.ssl.isEnabled()) { - settingsBuilder.applyToSslSettings(this::configureSsl); + if (this.connectionDetails != null) { + settingsBuilder.applyConnectionString(this.connectionDetails.getConnectionString()); + settingsBuilder.applyToSslSettings(this::configureSslIfNeeded); + } + else { + settingsBuilder.uuidRepresentation(this.uuidRepresentation); + settingsBuilder.applyConnectionString(this.connectionString); + if (this.ssl.isEnabled()) { + settingsBuilder.applyToSslSettings(this::configureSsl); + } } } @@ -73,6 +102,15 @@ private void configureSsl(SslSettings.Builder settings) { } } + private void configureSslIfNeeded(SslSettings.Builder settings) { + SslBundle sslBundle = this.connectionDetails.getSslBundle(); + if (sslBundle != null) { + settings.enabled(true); + Assert.state(!sslBundle.getOptions().isSpecified(), "SSL options cannot be specified with MongoDB"); + settings.context(sslBundle.createSslContext()); + } + } + @Override public int getOrder() { return this.order; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheProperties.java index 031c9b9e5ef1..004871d9496b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheProperties.java @@ -33,7 +33,7 @@ * @author Dave Syer * @since 1.2.2 */ -@ConfigurationProperties(prefix = "spring.mustache") +@ConfigurationProperties("spring.mustache") public class MustacheProperties { private static final MimeType DEFAULT_CONTENT_TYPE = MimeType.valueOf("text/html"); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheReactiveWebConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheReactiveWebConfiguration.java index eb27d957883d..6d52266d4d5b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheReactiveWebConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheReactiveWebConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,8 +18,8 @@ import com.samskivert.mustache.Mustache.Compiler; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.boot.context.properties.PropertyMapper; @@ -34,7 +34,7 @@ class MustacheReactiveWebConfiguration { @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(prefix = "spring.mustache", name = "enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.mustache.enabled", matchIfMissing = true) MustacheViewResolver mustacheViewResolver(Compiler mustacheCompiler, MustacheProperties mustache) { MustacheViewResolver resolver = new MustacheViewResolver(mustacheCompiler); PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull(); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheServletWebConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheServletWebConfiguration.java index 773c0f8550c2..cd7f01f312ec 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheServletWebConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/mustache/MustacheServletWebConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,9 +18,9 @@ import com.samskivert.mustache.Mustache.Compiler; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.boot.web.servlet.view.MustacheViewResolver; @@ -35,7 +35,7 @@ class MustacheServletWebConfiguration { @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(prefix = "spring.mustache", name = "enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.mustache.enabled", matchIfMissing = true) MustacheViewResolver mustacheViewResolver(Compiler mustacheCompiler, MustacheProperties mustache) { MustacheViewResolver resolver = new MustacheViewResolver(mustacheCompiler); resolver.setPrefix(mustache.getPrefix()); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/neo4j/Neo4jProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/neo4j/Neo4jProperties.java index 5b856a52e975..19e9df30fa92 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/neo4j/Neo4jProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/neo4j/Neo4jProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Stephane Nicoll * @since 2.4.0 */ -@ConfigurationProperties(prefix = "spring.neo4j") +@ConfigurationProperties("spring.neo4j") public class Neo4jProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/netty/NettyProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/netty/NettyProperties.java index de058a102ab0..619a59d469ab 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/netty/NettyProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/netty/NettyProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Brian Clozel * @since 2.5.0 */ -@ConfigurationProperties(prefix = "spring.netty") +@ConfigurationProperties("spring.netty") public class NettyProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaConfiguration.java index 8c9144b85ef7..6ab852375f43 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaConfiguration.java @@ -53,6 +53,7 @@ import org.springframework.boot.orm.jpa.hibernate.SpringJtaPlatform; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.ImportRuntimeHints; +import org.springframework.jdbc.support.SQLExceptionTranslator; import org.springframework.jndi.JndiLocatorDelegate; import org.springframework.orm.hibernate5.SpringBeanContainer; import org.springframework.orm.jpa.vendor.AbstractJpaVendorAdapter; @@ -95,6 +96,8 @@ class HibernateJpaConfiguration extends JpaBaseConfiguration { private final DataSourcePoolMetadataProvider poolMetadataProvider; + private final ObjectProvider sqlExceptionTranslator; + private final List hibernatePropertiesCustomizers; HibernateJpaConfiguration(DataSource dataSource, JpaProperties jpaProperties, @@ -104,11 +107,13 @@ class HibernateJpaConfiguration extends JpaBaseConfiguration { ObjectProvider providers, ObjectProvider physicalNamingStrategy, ObjectProvider implicitNamingStrategy, + ObjectProvider sqlExceptionTranslator, ObjectProvider hibernatePropertiesCustomizers) { super(dataSource, jpaProperties, jtaTransactionManager); this.hibernateProperties = hibernateProperties; this.defaultDdlAutoProvider = new HibernateDefaultDdlAutoProvider(providers); this.poolMetadataProvider = new CompositeDataSourcePoolMetadataProvider(metadataProviders.getIfAvailable()); + this.sqlExceptionTranslator = sqlExceptionTranslator; this.hibernatePropertiesCustomizers = determineHibernatePropertiesCustomizers( physicalNamingStrategy.getIfAvailable(), implicitNamingStrategy.getIfAvailable(), beanFactory, hibernatePropertiesCustomizers.orderedStream().toList()); @@ -134,7 +139,9 @@ private List determineHibernatePropertiesCustomiz @Override protected AbstractJpaVendorAdapter createJpaVendorAdapter() { - return new HibernateJpaVendorAdapter(); + HibernateJpaVendorAdapter adapter = new HibernateJpaVendorAdapter(); + this.sqlExceptionTranslator.ifUnique(adapter.getJpaDialect()::setJdbcExceptionTranslator); + return adapter; } @Override diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateProperties.java index c01450b1e55e..648ffd9c8340 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -77,8 +77,8 @@ public Naming getNaming() { */ public Map determineHibernateProperties(Map jpaProperties, HibernateSettings settings) { - Assert.notNull(jpaProperties, "JpaProperties must not be null"); - Assert.notNull(settings, "Settings must not be null"); + Assert.notNull(jpaProperties, "'jpaProperties' must not be null"); + Assert.notNull(settings, "'settings' must not be null"); return getAdditionalProperties(jpaProperties, settings); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/JpaBaseConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/JpaBaseConfiguration.java index 80306aaf906f..6d93dd71a660 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/JpaBaseConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/JpaBaseConfiguration.java @@ -30,9 +30,9 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfigurationPackages; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.boot.autoconfigure.domain.EntityScanPackages; @@ -159,7 +159,7 @@ public LocalContainerEntityManagerFactoryBean entityManagerFactory(EntityManager /** * Return the vendor-specific properties. * @return the vendor properties - * @deprecated since 3.4.4 for removal in 3.6.0 in favor of + * @deprecated since 3.4.4 for removal in 4.0.0 in favor of * {@link #getVendorProperties(DataSource)} */ @Deprecated(since = "3.4.4", forRemoval = true) @@ -241,7 +241,7 @@ private static String[] getPackagesToScan(BeanFactory beanFactory) { @ConditionalOnClass(WebMvcConfigurer.class) @ConditionalOnMissingBean({ OpenEntityManagerInViewInterceptor.class, OpenEntityManagerInViewFilter.class }) @ConditionalOnMissingFilterBean(OpenEntityManagerInViewFilter.class) - @ConditionalOnProperty(prefix = "spring.jpa", name = "open-in-view", havingValue = "true", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.jpa.open-in-view", matchIfMissing = true) protected static class JpaWebConfiguration { private static final Log logger = LogFactory.getLog(JpaWebConfiguration.class); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/JpaProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/JpaProperties.java index 9a208af28e7e..0e8baac795b6 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/JpaProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/JpaProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * @author Madhura Bhave * @since 1.1.0 */ -@ConfigurationProperties(prefix = "spring.jpa") +@ConfigurationProperties("spring.jpa") public class JpaProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarAutoConfiguration.java index a38a95f5a5db..f9a2144ae64c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,9 +28,9 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.thread.Threading; import org.springframework.boot.util.LambdaSafe; import org.springframework.context.annotation.Bean; @@ -87,7 +87,7 @@ public class PulsarAutoConfiguration { @Bean @ConditionalOnMissingBean(PulsarProducerFactory.class) - @ConditionalOnProperty(name = "spring.pulsar.producer.cache.enabled", havingValue = "false") + @ConditionalOnBooleanProperty(name = "spring.pulsar.producer.cache.enabled", havingValue = false) DefaultPulsarProducerFactory pulsarProducerFactory(PulsarClient pulsarClient, TopicResolver topicResolver, ObjectProvider> customizersProvider, ObjectProvider topicBuilderProvider) { @@ -101,7 +101,7 @@ DefaultPulsarProducerFactory pulsarProducerFactory(PulsarClient pulsarClient, @Bean @ConditionalOnMissingBean(PulsarProducerFactory.class) - @ConditionalOnProperty(name = "spring.pulsar.producer.cache.enabled", havingValue = "true", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.pulsar.producer.cache.enabled", matchIfMissing = true) CachingPulsarProducerFactory cachingPulsarProducerFactory(PulsarClient pulsarClient, TopicResolver topicResolver, ObjectProvider> customizersProvider, ObjectProvider topicBuilderProvider) { @@ -161,7 +161,7 @@ DefaultPulsarConsumerFactory pulsarConsumerFactory(PulsarClient pulsarClient, @Bean @ConditionalOnMissingBean(PulsarAwareTransactionManager.class) - @ConditionalOnProperty(prefix = "spring.pulsar.transaction", name = "enabled") + @ConditionalOnBooleanProperty("spring.pulsar.transaction.enabled") public PulsarTransactionManager pulsarTransactionManager(PulsarClient pulsarClient) { return new PulsarTransactionManager(pulsarClient); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarConfiguration.java index 2388fa4a6ee9..f7791c9241b1 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,8 +28,8 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.beans.factory.config.ConfigurableBeanFactory; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.pulsar.PulsarProperties.Defaults.SchemaInfo; import org.springframework.boot.autoconfigure.pulsar.PulsarProperties.Defaults.TypeMapping; import org.springframework.boot.context.properties.EnableConfigurationProperties; @@ -171,7 +171,7 @@ private void addCustomTopicMapping(DefaultTopicResolver topicResolver, TypeMappi @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(name = "spring.pulsar.function.enabled", havingValue = "true", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.pulsar.function.enabled", matchIfMissing = true) PulsarFunctionAdministration pulsarFunctionAdministration(PulsarAdministration pulsarAdministration, ObjectProvider pulsarFunctions, ObjectProvider pulsarSinks, ObjectProvider pulsarSources) { @@ -183,7 +183,7 @@ PulsarFunctionAdministration pulsarFunctionAdministration(PulsarAdministration p @Bean @Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE) @ConditionalOnMissingBean - @ConditionalOnProperty(name = "spring.pulsar.defaults.topic.enabled", havingValue = "true", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.pulsar.defaults.topic.enabled", matchIfMissing = true) PulsarTopicBuilder pulsarTopicBuilder() { return new PulsarTopicBuilder(TopicDomain.persistent, this.properties.getDefaults().getTopic().getTenant(), this.properties.getDefaults().getTopic().getNamespace()); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarProperties.java index 3972eec5029f..5ab34e4acfa4 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -292,9 +292,9 @@ public Topic getTopic() { public record TypeMapping(Class messageType, String topicName, SchemaInfo schemaInfo) { public TypeMapping { - Assert.notNull(messageType, "messageType must not be null"); + Assert.notNull(messageType, "'messageType' must not be null"); Assert.isTrue(topicName != null || schemaInfo != null, - "At least one of topicName or schemaInfo must not be null"); + "At least one of 'topicName' or 'schemaInfo' must not be null"); } } @@ -309,10 +309,10 @@ public record TypeMapping(Class messageType, String topicName, SchemaInfo sch public record SchemaInfo(SchemaType schemaType, Class messageKeyType) { public SchemaInfo { - Assert.notNull(schemaType, "schemaType must not be null"); - Assert.isTrue(schemaType != SchemaType.NONE, "schemaType 'NONE' not supported"); + Assert.notNull(schemaType, "'schemaType' must not be null"); + Assert.isTrue(schemaType != SchemaType.NONE, "'schemaType' must not be NONE"); Assert.isTrue(messageKeyType == null || schemaType == SchemaType.KEY_VALUE, - "messageKeyType can only be set when schemaType is KEY_VALUE"); + "'messageKeyType' can only be set when 'schemaType' is KEY_VALUE"); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarReactiveAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarReactiveAutoConfiguration.java index 45d8d3037212..07dfbc93ea3c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarReactiveAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/pulsar/PulsarReactiveAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,9 +33,9 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.util.LambdaSafe; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -90,7 +90,7 @@ ReactivePulsarClient reactivePulsarClient(PulsarClient pulsarClient) { @Bean @ConditionalOnMissingBean(ProducerCacheProvider.class) @ConditionalOnClass(CaffeineShadedProducerCacheProvider.class) - @ConditionalOnProperty(name = "spring.pulsar.producer.cache.enabled", havingValue = "true", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.pulsar.producer.cache.enabled", matchIfMissing = true) CaffeineShadedProducerCacheProvider reactivePulsarProducerCacheProvider() { PulsarProperties.Producer.Cache properties = this.properties.getProducer().getCache(); return new CaffeineShadedProducerCacheProvider(properties.getExpireAfterAccess(), Duration.ofMinutes(10), @@ -99,7 +99,7 @@ CaffeineShadedProducerCacheProvider reactivePulsarProducerCacheProvider() { @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(name = "spring.pulsar.producer.cache.enabled", havingValue = "true", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.pulsar.producer.cache.enabled", matchIfMissing = true) ReactiveMessageSenderCache reactivePulsarMessageSenderCache( ObjectProvider producerCacheProvider) { return reactivePulsarMessageSenderCache(producerCacheProvider.getIfAvailable()); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/quartz/QuartzAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/quartz/QuartzAutoConfiguration.java index 62567c53bf39..a9c5f4202bfb 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/quartz/QuartzAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/quartz/QuartzAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -97,7 +97,7 @@ private Properties asProperties(Map source) { @Configuration(proxyBeanMethods = false) @ConditionalOnSingleCandidate(DataSource.class) - @ConditionalOnProperty(prefix = "spring.quartz", name = "job-store-type", havingValue = "jdbc") + @ConditionalOnProperty(name = "spring.quartz.job-store-type", havingValue = "jdbc") @Import(DatabaseInitializationDependencyConfigurer.class) protected static class JdbcStoreTypeConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/ConnectionFactoryConfigurations.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/ConnectionFactoryConfigurations.java index 6807a349f77b..871304c50637 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/ConnectionFactoryConfigurations.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/ConnectionFactoryConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,9 +24,9 @@ import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.condition.ConditionOutcome; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.SpringBootCondition; import org.springframework.boot.autoconfigure.r2dbc.R2dbcProperties.Pool; import org.springframework.boot.context.properties.PropertyMapper; @@ -108,6 +108,7 @@ ConnectionPool connectionFactory(R2dbcProperties properties, map.from(pool.getMaxIdleTime()).to(builder::maxIdleTime); map.from(pool.getMaxLifeTime()).to(builder::maxLifeTime); map.from(pool.getMaxAcquireTime()).to(builder::maxAcquireTime); + map.from(pool.getAcquireRetry()).to(builder::acquireRetry); map.from(pool.getMaxCreateConnectionTime()).to(builder::maxCreateConnectionTime); map.from(pool.getInitialSize()).to(builder::initialSize); map.from(pool.getMaxSize()).to(builder::maxSize); @@ -123,8 +124,7 @@ ConnectionPool connectionFactory(R2dbcProperties properties, } @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.r2dbc.pool", value = "enabled", havingValue = "false", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.r2dbc.pool.enabled", havingValue = false, matchIfMissing = true) @ConditionalOnMissingBean(ConnectionFactory.class) static class GenericConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcProperties.java index e07b1dd0a06c..3591b54a3c09 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * @author Rodolpho S. Couto * @since 2.3.0 */ -@ConfigurationProperties(prefix = "spring.r2dbc") +@ConfigurationProperties("spring.r2dbc") public class R2dbcProperties { /** @@ -158,6 +158,11 @@ public static class Pool { */ private Duration maxAcquireTime; + /** + * Number of acquire retries if the first acquire attempt fails. + */ + private int acquireRetry = 1; + /** * Maximum time to validate a connection from the pool. By default, wait * indefinitely. @@ -234,6 +239,14 @@ public void setMaxAcquireTime(Duration maxAcquireTime) { this.maxAcquireTime = maxAcquireTime; } + public int getAcquireRetry() { + return this.acquireRetry; + } + + public void setAcquireRetry(int acquireRetry) { + this.acquireRetry = acquireRetry; + } + public Duration getMaxCreateConnectionTime() { return this.maxCreateConnectionTime; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/reactor/ReactorProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/reactor/ReactorProperties.java index c82da8b52389..7bbccbb79563 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/reactor/ReactorProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/reactor/ReactorProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,7 @@ * @author Brian Clozel * @since 3.2.0 */ -@ConfigurationProperties(prefix = "spring.reactor") +@ConfigurationProperties("spring.reactor") public class ReactorProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/reactor/netty/ReactorNettyProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/reactor/netty/ReactorNettyProperties.java index 4f085113c510..9b4a9422fdf1 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/reactor/netty/ReactorNettyProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/reactor/netty/ReactorNettyProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Moritz Halbritter * @since 2.7.9 */ -@ConfigurationProperties(prefix = "spring.reactor.netty") +@ConfigurationProperties("spring.reactor.netty") public class ReactorNettyProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/rsocket/RSocketServerAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/rsocket/RSocketServerAutoConfiguration.java index c9ae1d033847..70b62631ef32 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/rsocket/RSocketServerAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/rsocket/RSocketServerAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -94,7 +94,7 @@ private Consumer customizeWebsocketServerSpec(Spec spec) { } - @ConditionalOnProperty(prefix = "spring.rsocket.server", name = "port") + @ConditionalOnProperty("spring.rsocket.server.port") @ConditionalOnClass(ReactorResourceFactory.class) @Configuration(proxyBeanMethods = false) @Import(ReactorNettyConfigurations.ReactorResourceFactoryConfiguration.class) @@ -147,17 +147,17 @@ static class IsReactiveWebApplication { } - @ConditionalOnProperty(prefix = "spring.rsocket.server", name = "port", matchIfMissing = true) + @ConditionalOnProperty(name = "spring.rsocket.server.port", matchIfMissing = true) static class HasNoPortConfigured { } - @ConditionalOnProperty(prefix = "spring.rsocket.server", name = "mapping-path") + @ConditionalOnProperty("spring.rsocket.server.mapping-path") static class HasMappingPathConfigured { } - @ConditionalOnProperty(prefix = "spring.rsocket.server", name = "transport", havingValue = "websocket") + @ConditionalOnProperty(name = "spring.rsocket.server.transport", havingValue = "websocket") static class HasWebsocketTransportConfigured { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/SecurityProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/SecurityProperties.java index 504044c9eb98..69bcbe755764 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/SecurityProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/SecurityProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,7 +36,7 @@ * @author Madhura Bhave * @since 1.0.0 */ -@ConfigurationProperties(prefix = "spring.security") +@ConfigurationProperties("spring.security") public class SecurityProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/ClientsConfiguredCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/ClientsConfiguredCondition.java index d4d9df519ee2..e26c9b1c59f8 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/ClientsConfiguredCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/ClientsConfiguredCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,10 @@ * * @author Madhura Bhave * @since 2.1.0 + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of + * {@link ConditionalOnOAuth2ClientRegistrationProperties @ConditionalOnOAuth2ClientRegistrationConfigured} */ +@Deprecated(since = "3.5.0", forRemoval = true) public class ClientsConfiguredCondition extends SpringBootCondition { private static final Bindable> STRING_REGISTRATION_MAP = Bindable diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/ConditionalOnOAuth2ClientRegistrationProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/ConditionalOnOAuth2ClientRegistrationProperties.java new file mode 100644 index 000000000000..af76acda8c32 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/ConditionalOnOAuth2ClientRegistrationProperties.java @@ -0,0 +1,41 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.security.oauth2.client; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.context.annotation.Conditional; + +/** + * Condition that matches if any {@code spring.security.oauth2.client.registration} + * properties are defined. + * + * @author Andy Wilkinson + * @since 3.5.0 + */ +@SuppressWarnings("removal") +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.METHOD }) +@Documented +@Conditional(ClientsConfiguredCondition.class) +public @interface ConditionalOnOAuth2ClientRegistrationProperties { + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/OAuth2ClientProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/OAuth2ClientProperties.java index 48a30efc8cdc..65b1fbda8277 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/OAuth2ClientProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/OAuth2ClientProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * @author Moritz Halbritter * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.security.oauth2.client") +@ConfigurationProperties("spring.security.oauth2.client") public class OAuth2ClientProperties implements InitializingBean { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/reactive/ReactiveOAuth2ClientConfigurations.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/reactive/ReactiveOAuth2ClientConfigurations.java index 8eb3871b9377..8ab3b87a1831 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/reactive/ReactiveOAuth2ClientConfigurations.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/reactive/ReactiveOAuth2ClientConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,11 +22,10 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; -import org.springframework.boot.autoconfigure.security.oauth2.client.ClientsConfiguredCondition; +import org.springframework.boot.autoconfigure.security.oauth2.client.ConditionalOnOAuth2ClientRegistrationProperties; import org.springframework.boot.autoconfigure.security.oauth2.client.OAuth2ClientProperties; import org.springframework.boot.autoconfigure.security.oauth2.client.OAuth2ClientPropertiesMapper; import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Conditional; import org.springframework.context.annotation.Configuration; import org.springframework.security.config.web.server.ServerHttpSecurity; import org.springframework.security.oauth2.client.InMemoryReactiveOAuth2AuthorizedClientService; @@ -48,7 +47,7 @@ class ReactiveOAuth2ClientConfigurations { @Configuration(proxyBeanMethods = false) - @Conditional(ClientsConfiguredCondition.class) + @ConditionalOnOAuth2ClientRegistrationProperties @ConditionalOnMissingBean(ReactiveClientRegistrationRepository.class) static class ReactiveClientRegistrationRepositoryConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/servlet/OAuth2ClientRegistrationRepositoryConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/servlet/OAuth2ClientRegistrationRepositoryConfiguration.java index 09855253ad1a..edcb424d7f2f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/servlet/OAuth2ClientRegistrationRepositoryConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/client/servlet/OAuth2ClientRegistrationRepositoryConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,12 +20,11 @@ import java.util.List; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.security.oauth2.client.ClientsConfiguredCondition; +import org.springframework.boot.autoconfigure.security.oauth2.client.ConditionalOnOAuth2ClientRegistrationProperties; import org.springframework.boot.autoconfigure.security.oauth2.client.OAuth2ClientProperties; import org.springframework.boot.autoconfigure.security.oauth2.client.OAuth2ClientPropertiesMapper; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Conditional; import org.springframework.context.annotation.Configuration; import org.springframework.security.oauth2.client.registration.ClientRegistration; import org.springframework.security.oauth2.client.registration.ClientRegistrationRepository; @@ -39,7 +38,7 @@ */ @Configuration(proxyBeanMethods = false) @EnableConfigurationProperties(OAuth2ClientProperties.class) -@Conditional(ClientsConfiguredCondition.class) +@ConditionalOnOAuth2ClientRegistrationProperties class OAuth2ClientRegistrationRepositoryConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/ConditionalOnIssuerLocationJwtDecoder.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/ConditionalOnIssuerLocationJwtDecoder.java new file mode 100644 index 000000000000..2a8b8fe29ceb --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/ConditionalOnIssuerLocationJwtDecoder.java @@ -0,0 +1,42 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.security.oauth2.resource; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.context.annotation.Conditional; +import org.springframework.security.oauth2.jwt.NimbusJwtDecoder; + +/** + * Condition that matches when an {@link NimbusJwtDecoder#withIssuerLocation + * issuer-location-based JWT decoder} should be used. + * + * @author Andy Wilkinson + * @since 3.5.0 + */ +@SuppressWarnings("removal") +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.METHOD }) +@Documented +@Conditional(IssuerUriCondition.class) +public @interface ConditionalOnIssuerLocationJwtDecoder { + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/ConditionalOnPublicKeyJwtDecoder.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/ConditionalOnPublicKeyJwtDecoder.java new file mode 100644 index 000000000000..ef54bd012750 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/ConditionalOnPublicKeyJwtDecoder.java @@ -0,0 +1,42 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.security.oauth2.resource; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.context.annotation.Conditional; +import org.springframework.security.oauth2.jwt.NimbusJwtDecoder; + +/** + * Condition that matches when a {@link NimbusJwtDecoder#withPublicKey public-key-based + * JWT decoder} should be used. + * + * @author Andy Wilkinson + * @since 3.5.0 + */ +@SuppressWarnings("removal") +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.METHOD }) +@Documented +@Conditional(KeyValueCondition.class) +public @interface ConditionalOnPublicKeyJwtDecoder { + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/IssuerUriCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/IssuerUriCondition.java index d2e5c10763fc..688710b61530 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/IssuerUriCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/IssuerUriCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,7 +30,10 @@ * * @author Artsiom Yudovin * @since 2.1.0 + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of + * {@link ConditionalOnIssuerLocationJwtDecoder @ConditionalOnIssuerLocationJwtDecoder} */ +@Deprecated(since = "3.5.0", forRemoval = true) public class IssuerUriCondition extends SpringBootCondition { @Override @@ -38,10 +41,10 @@ public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeM ConditionMessage.Builder message = ConditionMessage.forCondition("OpenID Connect Issuer URI Condition"); Environment environment = context.getEnvironment(); String issuerUri = environment.getProperty("spring.security.oauth2.resourceserver.jwt.issuer-uri"); - String jwkSetUri = environment.getProperty("spring.security.oauth2.resourceserver.jwt.jwk-set-uri"); if (!StringUtils.hasText(issuerUri)) { return ConditionOutcome.noMatch(message.didNotFind("issuer-uri property").atAll()); } + String jwkSetUri = environment.getProperty("spring.security.oauth2.resourceserver.jwt.jwk-set-uri"); if (StringUtils.hasText(jwkSetUri)) { return ConditionOutcome.noMatch(message.found("jwk-set-uri property").items(jwkSetUri)); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/KeyValueCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/KeyValueCondition.java index 90b5d71cb0d7..e245484055ab 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/KeyValueCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/KeyValueCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,10 @@ * * @author Madhura Bhave * @since 2.2.0 + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of + * {@link ConditionalOnPublicKeyJwtDecoder @ConditionalOnPublicKeyJwtDecoder} */ +@Deprecated(since = "3.5.0", forRemoval = true) public class KeyValueCondition extends SpringBootCondition { @Override @@ -41,11 +44,11 @@ public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeM if (!StringUtils.hasText(publicKeyLocation)) { return ConditionOutcome.noMatch(message.didNotFind("public-key-location property").atAll()); } - String issuerUri = environment.getProperty("spring.security.oauth2.resourceserver.jwt.issuer-uri"); String jwkSetUri = environment.getProperty("spring.security.oauth2.resourceserver.jwt.jwk-set-uri"); if (StringUtils.hasText(jwkSetUri)) { return ConditionOutcome.noMatch(message.found("jwk-set-uri property").items(jwkSetUri)); } + String issuerUri = environment.getProperty("spring.security.oauth2.resourceserver.jwt.issuer-uri"); if (StringUtils.hasText(issuerUri)) { return ConditionOutcome.noMatch(message.found("issuer-uri property").items(issuerUri)); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/OAuth2ResourceServerProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/OAuth2ResourceServerProperties.java index f974fbbacfe2..48f1e2eca581 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/OAuth2ResourceServerProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/OAuth2ResourceServerProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,6 @@ import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.boot.context.properties.source.InvalidConfigurationPropertyValueException; import org.springframework.core.io.Resource; -import org.springframework.util.Assert; import org.springframework.util.StreamUtils; /** @@ -38,7 +37,7 @@ * @author Yan Kardziyaka * @since 2.1.0 */ -@ConfigurationProperties(prefix = "spring.security.oauth2.resourceserver") +@ConfigurationProperties("spring.security.oauth2.resourceserver") public class OAuth2ResourceServerProperties { private final Jwt jwt = new Jwt(); @@ -175,7 +174,10 @@ public void setPrincipalClaimName(String principalClaimName) { public String readPublicKey() throws IOException { String key = "spring.security.oauth2.resourceserver.public-key-location"; - Assert.notNull(this.publicKeyLocation, "PublicKeyLocation must not be null"); + if (this.publicKeyLocation == null) { + throw new InvalidConfigurationPropertyValueException(key, this.publicKeyLocation, + "No public key location specified"); + } if (!this.publicKeyLocation.exists()) { throw new InvalidConfigurationPropertyValueException(key, this.publicKeyLocation, "Public key location does not exist"); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/reactive/ReactiveOAuth2ResourceServerJwkConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/reactive/ReactiveOAuth2ResourceServerJwkConfiguration.java index 5d592a98e4be..2dbbb1476c73 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/reactive/ReactiveOAuth2ResourceServerJwkConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/reactive/ReactiveOAuth2ResourceServerJwkConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,8 +30,8 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; -import org.springframework.boot.autoconfigure.security.oauth2.resource.IssuerUriCondition; -import org.springframework.boot.autoconfigure.security.oauth2.resource.KeyValueCondition; +import org.springframework.boot.autoconfigure.security.oauth2.resource.ConditionalOnIssuerLocationJwtDecoder; +import org.springframework.boot.autoconfigure.security.oauth2.resource.ConditionalOnPublicKeyJwtDecoder; import org.springframework.boot.autoconfigure.security.oauth2.resource.OAuth2ResourceServerProperties; import org.springframework.boot.context.properties.PropertyMapper; import org.springframework.context.annotation.Bean; @@ -130,7 +130,7 @@ private boolean nullSafeDisjoint(List c1, List c2) { } @Bean - @Conditional(KeyValueCondition.class) + @ConditionalOnPublicKeyJwtDecoder NimbusReactiveJwtDecoder jwtDecoderByPublicKeyValue() throws Exception { RSAPublicKey publicKey = (RSAPublicKey) KeyFactory.getInstance("RSA") .generatePublic(new X509EncodedKeySpec(getKeySpec(this.properties.readPublicKey()))); @@ -158,7 +158,7 @@ private String exactlyOneAlgorithm() { } @Bean - @Conditional(IssuerUriCondition.class) + @ConditionalOnIssuerLocationJwtDecoder SupplierReactiveJwtDecoder jwtDecoderByIssuerUri( ObjectProvider customizers) { return new SupplierReactiveJwtDecoder(() -> { @@ -227,17 +227,17 @@ private static class JwtConverterPropertiesCondition extends AnyNestedCondition super(ConfigurationPhase.REGISTER_BEAN); } - @ConditionalOnProperty(prefix = "spring.security.oauth2.resourceserver.jwt", name = "authority-prefix") + @ConditionalOnProperty("spring.security.oauth2.resourceserver.jwt.authority-prefix") static class OnAuthorityPrefix { } - @ConditionalOnProperty(prefix = "spring.security.oauth2.resourceserver.jwt", name = "principal-claim-name") + @ConditionalOnProperty("spring.security.oauth2.resourceserver.jwt.principal-claim-name") static class OnPrincipalClaimName { } - @ConditionalOnProperty(prefix = "spring.security.oauth2.resourceserver.jwt", name = "authorities-claim-name") + @ConditionalOnProperty("spring.security.oauth2.resourceserver.jwt.authorities-claim-name") static class OnAuthoritiesClaimName { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/reactive/ReactiveOAuth2ResourceServerOpaqueTokenConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/reactive/ReactiveOAuth2ResourceServerOpaqueTokenConfiguration.java index f4d9614253e8..3fe7ea43ab12 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/reactive/ReactiveOAuth2ResourceServerOpaqueTokenConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/reactive/ReactiveOAuth2ResourceServerOpaqueTokenConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -46,8 +46,10 @@ static class OpaqueTokenIntrospectionClientConfiguration { @ConditionalOnProperty(name = "spring.security.oauth2.resourceserver.opaquetoken.introspection-uri") SpringReactiveOpaqueTokenIntrospector opaqueTokenIntrospector(OAuth2ResourceServerProperties properties) { OAuth2ResourceServerProperties.Opaquetoken opaqueToken = properties.getOpaquetoken(); - return new SpringReactiveOpaqueTokenIntrospector(opaqueToken.getIntrospectionUri(), - opaqueToken.getClientId(), opaqueToken.getClientSecret()); + return SpringReactiveOpaqueTokenIntrospector.withIntrospectionUri(opaqueToken.getIntrospectionUri()) + .clientId(opaqueToken.getClientId()) + .clientSecret(opaqueToken.getClientSecret()) + .build(); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/servlet/OAuth2ResourceServerJwtConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/servlet/OAuth2ResourceServerJwtConfiguration.java index f191a8173cc9..a4776565c910 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/servlet/OAuth2ResourceServerJwtConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/servlet/OAuth2ResourceServerJwtConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,8 +31,8 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.security.ConditionalOnDefaultWebSecurity; -import org.springframework.boot.autoconfigure.security.oauth2.resource.IssuerUriCondition; -import org.springframework.boot.autoconfigure.security.oauth2.resource.KeyValueCondition; +import org.springframework.boot.autoconfigure.security.oauth2.resource.ConditionalOnIssuerLocationJwtDecoder; +import org.springframework.boot.autoconfigure.security.oauth2.resource.ConditionalOnPublicKeyJwtDecoder; import org.springframework.boot.autoconfigure.security.oauth2.resource.OAuth2ResourceServerProperties; import org.springframework.boot.context.properties.PropertyMapper; import org.springframework.context.annotation.Bean; @@ -129,7 +129,7 @@ private boolean nullSafeDisjoint(List c1, List c2) { } @Bean - @Conditional(KeyValueCondition.class) + @ConditionalOnPublicKeyJwtDecoder JwtDecoder jwtDecoderByPublicKeyValue() throws Exception { RSAPublicKey publicKey = (RSAPublicKey) KeyFactory.getInstance("RSA") .generatePublic(new X509EncodedKeySpec(getKeySpec(this.properties.readPublicKey()))); @@ -157,7 +157,7 @@ private String exactlyOneAlgorithm() { } @Bean - @Conditional(IssuerUriCondition.class) + @ConditionalOnIssuerLocationJwtDecoder SupplierJwtDecoder jwtDecoderByIssuerUri(ObjectProvider customizers) { return new SupplierJwtDecoder(() -> { String issuerUri = this.properties.getIssuerUri(); @@ -219,17 +219,17 @@ private static class JwtConverterPropertiesCondition extends AnyNestedCondition super(ConfigurationPhase.REGISTER_BEAN); } - @ConditionalOnProperty(prefix = "spring.security.oauth2.resourceserver.jwt", name = "authority-prefix") + @ConditionalOnProperty("spring.security.oauth2.resourceserver.jwt.authority-prefix") static class OnAuthorityPrefix { } - @ConditionalOnProperty(prefix = "spring.security.oauth2.resourceserver.jwt", name = "principal-claim-name") + @ConditionalOnProperty("spring.security.oauth2.resourceserver.jwt.principal-claim-name") static class OnPrincipalClaimName { } - @ConditionalOnProperty(prefix = "spring.security.oauth2.resourceserver.jwt", name = "authorities-claim-name") + @ConditionalOnProperty("spring.security.oauth2.resourceserver.jwt.authorities-claim-name") static class OnAuthoritiesClaimName { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/servlet/OAuth2ResourceServerOpaqueTokenConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/servlet/OAuth2ResourceServerOpaqueTokenConfiguration.java index a995db7d896f..e2c28a88ade5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/servlet/OAuth2ResourceServerOpaqueTokenConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/resource/servlet/OAuth2ResourceServerOpaqueTokenConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,8 +48,10 @@ static class OpaqueTokenIntrospectionClientConfiguration { @ConditionalOnProperty(name = "spring.security.oauth2.resourceserver.opaquetoken.introspection-uri") SpringOpaqueTokenIntrospector opaqueTokenIntrospector(OAuth2ResourceServerProperties properties) { OAuth2ResourceServerProperties.Opaquetoken opaqueToken = properties.getOpaquetoken(); - return new SpringOpaqueTokenIntrospector(opaqueToken.getIntrospectionUri(), opaqueToken.getClientId(), - opaqueToken.getClientSecret()); + return SpringOpaqueTokenIntrospector.withIntrospectionUri(opaqueToken.getIntrospectionUri()) + .clientId(opaqueToken.getClientId()) + .clientSecret(opaqueToken.getClientSecret()) + .build(); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/server/servlet/OAuth2AuthorizationServerProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/server/servlet/OAuth2AuthorizationServerProperties.java index fd7ceba3c338..6df8a3d8444a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/server/servlet/OAuth2AuthorizationServerProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/oauth2/server/servlet/OAuth2AuthorizationServerProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * @author Steve Riesenberg * @since 3.1.0 */ -@ConfigurationProperties(prefix = "spring.security.oauth2.authorizationserver") +@ConfigurationProperties("spring.security.oauth2.authorizationserver") public class OAuth2AuthorizationServerProperties implements InitializingBean { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/reactive/ReactiveUserDetailsServiceAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/reactive/ReactiveUserDetailsServiceAutoConfiguration.java index 596f0d9c0b9d..f2e1332878a0 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/reactive/ReactiveUserDetailsServiceAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/reactive/ReactiveUserDetailsServiceAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -129,12 +129,12 @@ static final class MissingAlternative { } - @ConditionalOnProperty(prefix = "spring.security.user", name = "name") + @ConditionalOnProperty("spring.security.user.name") static final class NameConfigured { } - @ConditionalOnProperty(prefix = "spring.security.user", name = "password") + @ConditionalOnProperty("spring.security.user.password") static final class PasswordConfigured { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/reactive/StaticResourceRequest.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/reactive/StaticResourceRequest.java index 7d95c338c329..9f9321c047da 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/reactive/StaticResourceRequest.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/reactive/StaticResourceRequest.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -81,7 +81,7 @@ public StaticResourceServerWebExchange at(StaticResourceLocation first, StaticRe * @return the configured {@link ServerWebExchangeMatcher} */ public StaticResourceServerWebExchange at(Set locations) { - Assert.notNull(locations, "Locations must not be null"); + Assert.notNull(locations, "'locations' must not be null"); return new StaticResourceServerWebExchange(new LinkedHashSet<>(locations)); } @@ -115,7 +115,7 @@ public StaticResourceServerWebExchange excluding(StaticResourceLocation first, S * @return a new {@link StaticResourceServerWebExchange} */ public StaticResourceServerWebExchange excluding(Set locations) { - Assert.notNull(locations, "Locations must not be null"); + Assert.notNull(locations, "'locations' must not be null"); Set subset = new LinkedHashSet<>(this.locations); subset.removeAll(locations); return new StaticResourceServerWebExchange(subset); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/saml2/Saml2RelyingPartyRegistrationConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/saml2/Saml2RelyingPartyRegistrationConfiguration.java index aabed0af1bc6..5eab4a231a30 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/saml2/Saml2RelyingPartyRegistrationConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/saml2/Saml2RelyingPartyRegistrationConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -175,8 +175,8 @@ private RSAPrivateKey readPrivateKey(Resource location) { try (InputStream inputStream = location.getInputStream()) { PemContent pemContent = PemContent.load(inputStream); PrivateKey privateKey = pemContent.getPrivateKey(); - Assert.isInstanceOf(RSAPrivateKey.class, privateKey, - "PrivateKey in resource '" + location + "' must be an RSAPrivateKey"); + Assert.state(privateKey instanceof RSAPrivateKey, + () -> "PrivateKey in resource '" + location + "' must be an RSAPrivateKey"); return (RSAPrivateKey) privateKey; } catch (Exception ex) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/AntPathRequestMatcherProvider.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/AntPathRequestMatcherProvider.java index 2f6e9c0d7ed8..a9d776620bcf 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/AntPathRequestMatcherProvider.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/AntPathRequestMatcherProvider.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,10 @@ * * @author Madhura Bhave * @since 2.1.8 + * @deprecated since 3.5.0 for removal in 4.0.0 along with {@link RequestMatcherProvider} */ +@Deprecated(since = "3.5.0", forRemoval = true) +@SuppressWarnings("removal") public class AntPathRequestMatcherProvider implements RequestMatcherProvider { private final Function pathFactory; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/RequestMatcherProvider.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/RequestMatcherProvider.java index 39c20aeb1534..4a7aeb2491ec 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/RequestMatcherProvider.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/RequestMatcherProvider.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,10 @@ * * @author Madhura Bhave * @since 2.0.5 + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of + * {@code org.springframework.boot.actuate.autoconfigure.security.servlet.RequestMatcherProvider} */ +@Deprecated(since = "3.5.0", forRemoval = true) @FunctionalInterface public interface RequestMatcherProvider { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/StaticResourceRequest.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/StaticResourceRequest.java index 13d6cf92681c..ef8a72bc5cd5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/StaticResourceRequest.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/StaticResourceRequest.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -86,7 +86,7 @@ public StaticResourceRequestMatcher at(StaticResourceLocation first, StaticResou * @return the configured {@link RequestMatcher} */ public StaticResourceRequestMatcher at(Set locations) { - Assert.notNull(locations, "Locations must not be null"); + Assert.notNull(locations, "'locations' must not be null"); return new StaticResourceRequestMatcher(new LinkedHashSet<>(locations)); } @@ -124,7 +124,7 @@ public StaticResourceRequestMatcher excluding(StaticResourceLocation first, Stat * @return a new {@link StaticResourceRequestMatcher} */ public StaticResourceRequestMatcher excluding(Set locations) { - Assert.notNull(locations, "Locations must not be null"); + Assert.notNull(locations, "'locations' must not be null"); Set subset = new LinkedHashSet<>(this.locations); subset.removeAll(locations); return new StaticResourceRequestMatcher(subset); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/UserDetailsServiceAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/UserDetailsServiceAutoConfiguration.java index a6b803083720..365c1b8b2b4b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/UserDetailsServiceAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/servlet/UserDetailsServiceAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -113,12 +113,12 @@ static final class MissingAlternative { } - @ConditionalOnProperty(prefix = "spring.security.user", name = "name") + @ConditionalOnProperty("spring.security.user.name") static final class NameConfigured { } - @ConditionalOnProperty(prefix = "spring.security.user", name = "password") + @ConditionalOnProperty("spring.security.user.password") static final class PasswordConfigured { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sendgrid/SendGridAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sendgrid/SendGridAutoConfiguration.java index b51ad0610390..a32f16ee1b23 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sendgrid/SendGridAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sendgrid/SendGridAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,7 +40,7 @@ */ @AutoConfiguration @ConditionalOnClass(SendGrid.class) -@ConditionalOnProperty(prefix = "spring.sendgrid", value = "api-key") +@ConditionalOnProperty("spring.sendgrid.api-key") @EnableConfigurationProperties(SendGridProperties.class) public class SendGridAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sendgrid/SendGridProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sendgrid/SendGridProperties.java index 705978a46a62..3c68d7bca9f4 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sendgrid/SendGridProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sendgrid/SendGridProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ * @author Andy Wilkinson * @since 1.3.0 */ -@ConfigurationProperties(prefix = "spring.sendgrid") +@ConfigurationProperties("spring.sendgrid") public class SendGridProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/HazelcastSessionProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/HazelcastSessionProperties.java index 9cb4814c8671..b51a626a590b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/HazelcastSessionProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/HazelcastSessionProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Vedran Pavic * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.session.hazelcast") +@ConfigurationProperties("spring.session.hazelcast") public class HazelcastSessionProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/JdbcSessionProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/JdbcSessionProperties.java index 80fedbdf1100..07c15888f5dc 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/JdbcSessionProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/JdbcSessionProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ * @author Vedran Pavic * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.session.jdbc") +@ConfigurationProperties("spring.session.jdbc") public class JdbcSessionProperties { private static final String DEFAULT_SCHEMA_LOCATION = "classpath:org/springframework/" diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/MongoSessionProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/MongoSessionProperties.java index e1c844beb01f..17b7c9068f77 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/MongoSessionProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/MongoSessionProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,7 @@ * @author Andy Wilkinson * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.session.mongodb") +@ConfigurationProperties("spring.session.mongodb") public class MongoSessionProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisReactiveSessionConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisReactiveSessionConfiguration.java index b380d495088c..e2ee0c7f50c2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisReactiveSessionConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisReactiveSessionConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -51,7 +51,7 @@ class RedisReactiveSessionConfiguration { @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.session.redis", name = "repository-type", havingValue = "default", + @ConditionalOnProperty(name = "spring.session.redis.repository-type", havingValue = "default", matchIfMissing = true) @Import(RedisWebSessionConfiguration.class) static class DefaultRedisSessionConfiguration { @@ -73,7 +73,7 @@ ReactiveSessionRepositoryCustomizer springBootSe } @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.session.redis", name = "repository-type", havingValue = "indexed") + @ConditionalOnProperty(name = "spring.session.redis.repository-type", havingValue = "indexed") @Import(RedisIndexedWebSessionConfiguration.class) static class IndexedRedisSessionConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisSessionConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisSessionConfiguration.java index 71faf7798713..659997010de9 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisSessionConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisSessionConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -57,7 +57,7 @@ class RedisSessionConfiguration { @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.session.redis", name = "repository-type", havingValue = "default", + @ConditionalOnProperty(name = "spring.session.redis.repository-type", havingValue = "default", matchIfMissing = true) @Import(RedisHttpSessionConfiguration.class) static class DefaultRedisSessionConfiguration { @@ -87,7 +87,7 @@ SessionRepositoryCustomizer springBootSessionRepositoryC } @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.session.redis", name = "repository-type", havingValue = "indexed") + @ConditionalOnProperty(name = "spring.session.redis.repository-type", havingValue = "indexed") @Import(RedisIndexedHttpSessionConfiguration.class) static class IndexedRedisSessionConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisSessionProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisSessionProperties.java index 93f16dc5ca1f..d2604b8bb3bb 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisSessionProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/RedisSessionProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Vedran Pavic * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.session.redis") +@ConfigurationProperties("spring.session.redis") public class RedisSessionProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/SessionProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/SessionProperties.java index 962621274655..452d11c71a61 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/SessionProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/session/SessionProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,7 +36,7 @@ * @author Vedran Pavic * @since 1.4.0 */ -@ConfigurationProperties(prefix = "spring.session") +@ConfigurationProperties("spring.session") public class SessionProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sql/init/OnDatabaseInitializationCondition.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sql/init/OnDatabaseInitializationCondition.java index 032a04731479..0cdf101fbff3 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sql/init/OnDatabaseInitializationCondition.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sql/init/OnDatabaseInitializationCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ * @since 2.6.2 * @see DatabaseInitializationMode */ -public class OnDatabaseInitializationCondition extends SpringBootCondition { +public abstract class OnDatabaseInitializationCondition extends SpringBootCondition { private final String name; @@ -48,7 +48,7 @@ public class OnDatabaseInitializationCondition extends SpringBootCondition { * @param name the name of the component * @param propertyNames the properties to check (in order) */ - public OnDatabaseInitializationCondition(String name, String... propertyNames) { + protected OnDatabaseInitializationCondition(String name, String... propertyNames) { this.name = name; this.propertyNames = propertyNames; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sql/init/SqlInitializationAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sql/init/SqlInitializationAutoConfiguration.java index 09f4bf71ee9e..c249a581cccf 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sql/init/SqlInitializationAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/sql/init/SqlInitializationAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,6 +18,7 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.NoneNestedConditions; import org.springframework.boot.autoconfigure.sql.init.SqlInitializationAutoConfiguration.SqlInitializationModeCondition; @@ -36,7 +37,7 @@ @EnableConfigurationProperties(SqlInitializationProperties.class) @Import({ DatabaseInitializationDependencyConfigurer.class, R2dbcInitializationConfiguration.class, DataSourceInitializationConfiguration.class }) -@ConditionalOnProperty(prefix = "spring.sql.init", name = "enabled", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.sql.init.enabled", matchIfMissing = true) @Conditional(SqlInitializationModeCondition.class) public class SqlInitializationAutoConfiguration { @@ -46,7 +47,7 @@ static class SqlInitializationModeCondition extends NoneNestedConditions { super(ConfigurationPhase.PARSE_CONFIGURATION); } - @ConditionalOnProperty(prefix = "spring.sql.init", name = "mode", havingValue = "never") + @ConditionalOnProperty(name = "spring.sql.init.mode", havingValue = "never") static class ModeIsNever { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/CertificateMatcher.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/CertificateMatcher.java index 3f25ecc2c0c4..49604a466fcf 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/CertificateMatcher.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/CertificateMatcher.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -50,10 +50,10 @@ class CertificateMatcher { private final byte[] generatedSignature; CertificateMatcher(PrivateKey privateKey) { - Assert.notNull(privateKey, "Private key must not be null"); + Assert.notNull(privateKey, "'privateKey' must not be null"); this.privateKey = privateKey; this.signature = createSignature(privateKey); - Assert.notNull(this.signature, "Failed to create signature"); + Assert.state(this.signature != null, "Failed to create signature"); this.generatedSignature = sign(this.signature, privateKey); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/FileWatcher.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/FileWatcher.java index ba05ecd28dde..449dcccabd2e 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/FileWatcher.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/FileWatcher.java @@ -35,7 +35,6 @@ import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.CopyOnWriteArrayList; import java.util.concurrent.TimeUnit; -import java.util.stream.Collectors; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; @@ -65,7 +64,7 @@ class FileWatcher implements Closeable { * actions */ FileWatcher(Duration quietPeriod) { - Assert.notNull(quietPeriod, "QuietPeriod must not be null"); + Assert.notNull(quietPeriod, "'quietPeriod' must not be null"); this.quietPeriod = quietPeriod; } @@ -75,8 +74,8 @@ class FileWatcher implements Closeable { * @param action the action to take when changes are detected */ void watch(Set paths, Runnable action) { - Assert.notNull(paths, "Paths must not be null"); - Assert.notNull(action, "Action must not be null"); + Assert.notNull(paths, "'paths' must not be null"); + Assert.notNull(action, "'action' must not be null"); if (paths.isEmpty()) { return; } @@ -86,11 +85,11 @@ void watch(Set paths, Runnable action) { this.thread = new WatcherThread(); this.thread.start(); } - Set actualPaths = new HashSet<>(); + Set registrationPaths = new HashSet<>(); for (Path path : paths) { - actualPaths.add(resolveSymlinkIfNecessary(path)); + registrationPaths.addAll(getRegistrationPaths(path)); } - this.thread.register(new Registration(actualPaths, action)); + this.thread.register(new Registration(registrationPaths, action)); } catch (IOException ex) { throw new UncheckedIOException("Failed to register paths for watching: " + paths, ex); @@ -98,14 +97,6 @@ void watch(Set paths, Runnable action) { } } - private static Path resolveSymlinkIfNecessary(Path path) throws IOException { - if (Files.isSymbolicLink(path)) { - Path target = path.resolveSibling(Files.readSymbolicLink(path)); - return resolveSymlinkIfNecessary(target); - } - return path; - } - @Override public void close() throws IOException { synchronized (this.lock) { @@ -123,6 +114,44 @@ public void close() throws IOException { } } + /** + * Retrieves all {@link Path Paths} that should be registered for the specified + * {@link Path}. If the path is a symlink, changes to the symlink should be monitored, + * not just the file it points to. For example, for the given {@code keystore.jks} + * path in the following directory structure:
+	 * .
+	 * ├── ..a72e81ff-f0e1-41d8-a19b-068d3d1d4e2f
+	 * │   ├── keystore.jks
+	 * ├── ..data -> ..a72e81ff-f0e1-41d8-a19b-068d3d1d4e2f
+	 * ├── keystore.jks -> ..data/keystore.jks
+	 * 
the resulting paths would include: + *
    + *
  • keystore.jks
  • + *
  • ..data/keystore.jks
  • + *
  • ..data
  • + *
  • ..a72e81ff-f0e1-41d8-a19b-068d3d1d4e2f/keystore.jks
  • + *
+ * @param path the path + * @return all possible {@link Path} instances to be registered + * @throws IOException if an I/O error occurs + */ + private static Set getRegistrationPaths(Path path) throws IOException { + path = path.toAbsolutePath(); + Set result = new HashSet<>(); + result.add(path); + Path parent = path.getParent(); + if (parent != null && Files.isSymbolicLink(parent)) { + result.add(parent); + Path target = parent.resolveSibling(Files.readSymbolicLink(parent)); + result.addAll(getRegistrationPaths(target.resolve(path.getFileName()))); + } + else if (Files.isSymbolicLink(path)) { + Path target = path.resolveSibling(Files.readSymbolicLink(path)); + result.addAll(getRegistrationPaths(target)); + } + return result; + } + /** * The watcher thread used to check for changes. */ @@ -145,11 +174,15 @@ private void onThreadException(Thread thread, Throwable throwable) { } void register(Registration registration) throws IOException { + Set directories = new HashSet<>(); for (Path path : registration.paths()) { if (!Files.isRegularFile(path) && !Files.isDirectory(path)) { throw new IOException("'%s' is neither a file nor a directory".formatted(path)); } Path directory = Files.isDirectory(path) ? path : path.getParent(); + directories.add(directory); + } + for (Path directory : directories) { WatchKey watchKey = register(directory); this.registrations.computeIfAbsent(watchKey, (key) -> new CopyOnWriteArrayList<>()).add(registration); } @@ -224,10 +257,6 @@ public void close() throws IOException { */ private record Registration(Set paths, Runnable action) { - Registration { - paths = paths.stream().map(Path::toAbsolutePath).collect(Collectors.toSet()); - } - boolean manages(Path file) { Path absolutePath = file.toAbsolutePath(); return this.paths.contains(absolutePath) || isInDirectories(absolutePath); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/PropertiesSslBundle.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/PropertiesSslBundle.java index 1a8a35aa4e0a..dc42c7ecb053 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/PropertiesSslBundle.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/PropertiesSslBundle.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -127,7 +127,7 @@ private static PemSslStore getPemSslStore(String propertyName, PemSslBundlePrope if (properties.isVerifyKeys()) { CertificateMatcher certificateMatcher = new CertificateMatcher(pemSslStore.privateKey()); Assert.state(certificateMatcher.matchesAny(pemSslStore.certificates()), - "Private key in %s matches none of the certificates in the chain".formatted(propertyName)); + () -> "Private key in %s matches none of the certificates in the chain".formatted(propertyName)); } return pemSslStore; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/SslProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/SslProperties.java index a755a871b8d7..59bca669fc05 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/SslProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/SslProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Moritz Halbritter * @since 3.1.0 */ -@ConfigurationProperties(prefix = "spring.ssl") +@ConfigurationProperties("spring.ssl") public class SslProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskExecutionProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskExecutionProperties.java index 4f30ab25de33..759d336bc235 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskExecutionProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskExecutionProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,6 +37,11 @@ public class TaskExecutionProperties { private final Shutdown shutdown = new Shutdown(); + /** + * Determine when the task executor is to be created. + */ + private Mode mode = Mode.AUTO; + /** * Prefix to use for the names of newly created threads. */ @@ -54,6 +59,14 @@ public Shutdown getShutdown() { return this.shutdown; } + public Mode getMode() { + return this.mode; + } + + public void setMode(Mode mode) { + this.mode = mode; + } + public String getThreadNamePrefix() { return this.threadNamePrefix; } @@ -209,4 +222,23 @@ public void setAwaitTerminationPeriod(Duration awaitTerminationPeriod) { } + /** + * Determine when the task executor is to be created. + * + * @since 3.5.0 + */ + public enum Mode { + + /** + * Create the task executor if no user-defined executor is present. + */ + AUTO, + + /** + * Create the task executor even if a user-defined executor is present. + */ + FORCE + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskExecutorConfigurations.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskExecutorConfigurations.java index ee2bbf14e843..ffecc0144210 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskExecutorConfigurations.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskExecutorConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,8 +18,11 @@ import java.util.concurrent.Executor; +import org.springframework.beans.factory.BeanFactory; import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.autoconfigure.condition.AnyNestedCondition; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnThreading; import org.springframework.boot.autoconfigure.thread.Threading; import org.springframework.boot.task.SimpleAsyncTaskExecutorBuilder; @@ -27,12 +30,14 @@ import org.springframework.boot.task.ThreadPoolTaskExecutorBuilder; import org.springframework.boot.task.ThreadPoolTaskExecutorCustomizer; import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Conditional; import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Import; import org.springframework.context.annotation.Lazy; import org.springframework.core.task.SimpleAsyncTaskExecutor; import org.springframework.core.task.TaskDecorator; import org.springframework.core.task.TaskExecutor; -import org.springframework.scheduling.annotation.AsyncAnnotationBeanPostProcessor; +import org.springframework.scheduling.annotation.AsyncConfigurer; import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; /** @@ -46,19 +51,18 @@ class TaskExecutorConfigurations { @Configuration(proxyBeanMethods = false) - @ConditionalOnMissingBean(Executor.class) + @Conditional(OnExecutorCondition.class) + @Import(AsyncConfigurerConfiguration.class) static class TaskExecutorConfiguration { - @Bean(name = { TaskExecutionAutoConfiguration.APPLICATION_TASK_EXECUTOR_BEAN_NAME, - AsyncAnnotationBeanPostProcessor.DEFAULT_TASK_EXECUTOR_BEAN_NAME }) + @Bean(TaskExecutionAutoConfiguration.APPLICATION_TASK_EXECUTOR_BEAN_NAME) @ConditionalOnThreading(Threading.VIRTUAL) SimpleAsyncTaskExecutor applicationTaskExecutorVirtualThreads(SimpleAsyncTaskExecutorBuilder builder) { return builder.build(); } + @Bean(TaskExecutionAutoConfiguration.APPLICATION_TASK_EXECUTOR_BEAN_NAME) @Lazy - @Bean(name = { TaskExecutionAutoConfiguration.APPLICATION_TASK_EXECUTOR_BEAN_NAME, - AsyncAnnotationBeanPostProcessor.DEFAULT_TASK_EXECUTOR_BEAN_NAME }) @ConditionalOnThreading(Threading.PLATFORM) ThreadPoolTaskExecutor applicationTaskExecutor(ThreadPoolTaskExecutorBuilder threadPoolTaskExecutorBuilder) { return threadPoolTaskExecutorBuilder.build(); @@ -140,4 +144,40 @@ private SimpleAsyncTaskExecutorBuilder builder() { } + @Configuration(proxyBeanMethods = false) + @ConditionalOnMissingBean(AsyncConfigurer.class) + static class AsyncConfigurerConfiguration { + + @Bean + @ConditionalOnMissingBean + AsyncConfigurer applicationTaskExecutorAsyncConfigurer(BeanFactory beanFactory) { + return new AsyncConfigurer() { + @Override + public Executor getAsyncExecutor() { + return beanFactory.getBean(TaskExecutionAutoConfiguration.APPLICATION_TASK_EXECUTOR_BEAN_NAME, + Executor.class); + } + }; + } + + } + + static class OnExecutorCondition extends AnyNestedCondition { + + OnExecutorCondition() { + super(ConfigurationPhase.REGISTER_BEAN); + } + + @ConditionalOnMissingBean(Executor.class) + private static final class ExecutorBeanCondition { + + } + + @ConditionalOnProperty(value = "spring.task.execution.mode", havingValue = "force") + private static final class ModelCondition { + + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskSchedulingConfigurations.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskSchedulingConfigurations.java index 0112171fa7af..23f654409230 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskSchedulingConfigurations.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/task/TaskSchedulingConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,6 +29,7 @@ import org.springframework.boot.task.ThreadPoolTaskSchedulerCustomizer; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.core.task.TaskDecorator; import org.springframework.scheduling.TaskScheduler; import org.springframework.scheduling.concurrent.SimpleAsyncTaskScheduler; import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler; @@ -67,6 +68,7 @@ static class ThreadPoolTaskSchedulerBuilderConfiguration { @Bean @ConditionalOnMissingBean ThreadPoolTaskSchedulerBuilder threadPoolTaskSchedulerBuilder(TaskSchedulingProperties properties, + ObjectProvider taskDecorator, ObjectProvider threadPoolTaskSchedulerCustomizers) { TaskSchedulingProperties.Shutdown shutdown = properties.getShutdown(); ThreadPoolTaskSchedulerBuilder builder = new ThreadPoolTaskSchedulerBuilder(); @@ -74,6 +76,7 @@ ThreadPoolTaskSchedulerBuilder threadPoolTaskSchedulerBuilder(TaskSchedulingProp builder = builder.awaitTermination(shutdown.isAwaitTermination()); builder = builder.awaitTerminationPeriod(shutdown.getAwaitTerminationPeriod()); builder = builder.threadNamePrefix(properties.getThreadNamePrefix()); + builder = builder.taskDecorator(taskDecorator.getIfUnique()); builder = builder.customizers(threadPoolTaskSchedulerCustomizers); return builder; } @@ -85,11 +88,15 @@ static class SimpleAsyncTaskSchedulerBuilderConfiguration { private final TaskSchedulingProperties properties; + private final ObjectProvider taskDecorator; + private final ObjectProvider taskSchedulerCustomizers; SimpleAsyncTaskSchedulerBuilderConfiguration(TaskSchedulingProperties properties, + ObjectProvider taskDecorator, ObjectProvider taskSchedulerCustomizers) { this.properties = properties; + this.taskDecorator = taskDecorator; this.taskSchedulerCustomizers = taskSchedulerCustomizers; } @@ -110,6 +117,7 @@ SimpleAsyncTaskSchedulerBuilder simpleAsyncTaskSchedulerBuilderVirtualThreads() private SimpleAsyncTaskSchedulerBuilder builder() { SimpleAsyncTaskSchedulerBuilder builder = new SimpleAsyncTaskSchedulerBuilder(); builder = builder.threadNamePrefix(this.properties.getThreadNamePrefix()); + builder = builder.taskDecorator(this.taskDecorator.getIfUnique()); builder = builder.customizers(this.taskSchedulerCustomizers.orderedStream()::iterator); TaskSchedulingProperties.Simple simple = this.properties.getSimple(); builder = builder.concurrencyLimit(simple.getConcurrencyLimit()); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/template/TemplateAvailabilityProviders.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/template/TemplateAvailabilityProviders.java index f6e2d678fb04..d20aff3900e6 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/template/TemplateAvailabilityProviders.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/template/TemplateAvailabilityProviders.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -80,7 +80,7 @@ public TemplateAvailabilityProviders(ApplicationContext applicationContext) { * @param classLoader the source class loader */ public TemplateAvailabilityProviders(ClassLoader classLoader) { - Assert.notNull(classLoader, "ClassLoader must not be null"); + Assert.notNull(classLoader, "'classLoader' must not be null"); this.providers = SpringFactoriesLoader.loadFactories(TemplateAvailabilityProvider.class, classLoader); } @@ -89,7 +89,7 @@ public TemplateAvailabilityProviders(ClassLoader classLoader) { * @param providers the underlying providers */ protected TemplateAvailabilityProviders(Collection providers) { - Assert.notNull(providers, "Providers must not be null"); + Assert.notNull(providers, "'providers' must not be null"); this.providers = new ArrayList<>(providers); } @@ -108,7 +108,7 @@ public List getProviders() { * @return a {@link TemplateAvailabilityProvider} or null */ public TemplateAvailabilityProvider getProvider(String view, ApplicationContext applicationContext) { - Assert.notNull(applicationContext, "ApplicationContext must not be null"); + Assert.notNull(applicationContext, "'applicationContext' must not be null"); return getProvider(view, applicationContext.getEnvironment(), applicationContext.getClassLoader(), applicationContext); } @@ -123,10 +123,10 @@ public TemplateAvailabilityProvider getProvider(String view, ApplicationContext */ public TemplateAvailabilityProvider getProvider(String view, Environment environment, ClassLoader classLoader, ResourceLoader resourceLoader) { - Assert.notNull(view, "View must not be null"); - Assert.notNull(environment, "Environment must not be null"); - Assert.notNull(classLoader, "ClassLoader must not be null"); - Assert.notNull(resourceLoader, "ResourceLoader must not be null"); + Assert.notNull(view, "'view' must not be null"); + Assert.notNull(environment, "'environment' must not be null"); + Assert.notNull(classLoader, "'classLoader' must not be null"); + Assert.notNull(resourceLoader, "'resourceLoader' must not be null"); Boolean useCache = environment.getProperty("spring.template.provider.cache", Boolean.class, true); if (!useCache) { return findProvider(view, environment, classLoader, resourceLoader); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/template/TemplateLocation.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/template/TemplateLocation.java index 90dd2caf135a..a5adcc4881ea 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/template/TemplateLocation.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/template/TemplateLocation.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ public class TemplateLocation { private final String path; public TemplateLocation(String path) { - Assert.notNull(path, "Path must not be null"); + Assert.notNull(path, "'path' must not be null"); this.path = path; } @@ -45,7 +45,7 @@ public TemplateLocation(String path) { * @return {@code true} if the location exists. */ public boolean exists(ResourcePatternResolver resolver) { - Assert.notNull(resolver, "Resolver must not be null"); + Assert.notNull(resolver, "'resolver' must not be null"); if (resolver.getResource(this.path).exists()) { return true; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/TemplateEngineConfigurations.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/TemplateEngineConfigurations.java index 62bac976e196..23de094eb336 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/TemplateEngineConfigurations.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/TemplateEngineConfigurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,8 +25,8 @@ import org.thymeleaf.templateresolver.ITemplateResolver; import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.context.annotation.Bean; @@ -59,7 +59,7 @@ SpringTemplateEngine templateEngine(ThymeleafProperties properties, @Configuration(proxyBeanMethods = false) @ConditionalOnWebApplication(type = Type.REACTIVE) - @ConditionalOnProperty(name = "spring.thymeleaf.enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.thymeleaf.enabled", matchIfMissing = true) static class ReactiveTemplateEngineConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/ThymeleafAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/ThymeleafAutoConfiguration.java index 7a088eb8af99..25c20208ca56 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/ThymeleafAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/ThymeleafAutoConfiguration.java @@ -33,9 +33,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.boot.autoconfigure.template.TemplateLocation; @@ -129,7 +129,7 @@ SpringResourceTemplateResolver defaultTemplateResolver() { @Configuration(proxyBeanMethods = false) @ConditionalOnWebApplication(type = Type.SERVLET) - @ConditionalOnProperty(name = "spring.thymeleaf.enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.thymeleaf.enabled", matchIfMissing = true) static class ThymeleafWebMvcConfiguration { @Bean @@ -182,7 +182,7 @@ private String appendCharset(MimeType type, String charset) { @Configuration(proxyBeanMethods = false) @ConditionalOnWebApplication(type = Type.REACTIVE) - @ConditionalOnProperty(name = "spring.thymeleaf.enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.thymeleaf.enabled", matchIfMissing = true) static class ThymeleafWebFluxConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/ThymeleafProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/ThymeleafProperties.java index b1d32a5a2c25..277d9e9345c5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/ThymeleafProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/thymeleaf/ThymeleafProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * @author Kazuki Shimizu * @since 1.2.0 */ -@ConfigurationProperties(prefix = "spring.thymeleaf") +@ConfigurationProperties("spring.thymeleaf") public class ThymeleafProperties { private static final Charset DEFAULT_ENCODING = StandardCharsets.UTF_8; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/TransactionAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/TransactionAutoConfiguration.java index b1268020c454..ecdb15452e85 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/TransactionAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/TransactionAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,9 +19,9 @@ import org.springframework.boot.LazyInitializationExcludeFilter; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnSingleCandidate; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -72,15 +72,14 @@ public static class EnableTransactionManagementConfiguration { @Configuration(proxyBeanMethods = false) @EnableTransactionManagement(proxyTargetClass = false) - @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class", havingValue = "false") + @ConditionalOnBooleanProperty(name = "spring.aop.proxy-target-class", havingValue = false) public static class JdkDynamicAutoProxyConfiguration { } @Configuration(proxyBeanMethods = false) @EnableTransactionManagement(proxyTargetClass = true) - @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class", havingValue = "true", - matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.aop.proxy-target-class", matchIfMissing = true) public static class CglibAutoProxyConfiguration { } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/TransactionProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/TransactionProperties.java index a9170ee0cd78..1a9fb9054802 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/TransactionProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/TransactionProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ * @author Phillip Webb * @since 1.5.0 */ -@ConfigurationProperties(prefix = "spring.transaction") +@ConfigurationProperties("spring.transaction") public class TransactionProperties implements TransactionManagerCustomizer { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/jta/JtaAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/jta/JtaAutoConfiguration.java index 480e32b81dfa..677002e03e1d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/jta/JtaAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/transaction/jta/JtaAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,8 +18,8 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.jdbc.XADataSourceAutoConfiguration; import org.springframework.boot.autoconfigure.jms.activemq.ActiveMQAutoConfiguration; import org.springframework.boot.autoconfigure.jms.artemis.ArtemisAutoConfiguration; @@ -40,7 +40,7 @@ ArtemisAutoConfiguration.class, HibernateJpaAutoConfiguration.class, TransactionAutoConfiguration.class, TransactionManagerCustomizationAutoConfiguration.class }) @ConditionalOnClass(jakarta.transaction.Transaction.class) -@ConditionalOnProperty(prefix = "spring.jta", value = "enabled", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.jta.enabled", matchIfMissing = true) @Import(JndiJtaConfiguration.class) public class JtaAutoConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/validation/ValidationAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/validation/ValidationAutoConfiguration.java index a3df37fb1188..d1af26e17a05 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/validation/ValidationAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/validation/ValidationAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,6 +27,7 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnResource; import org.springframework.boot.autoconfigure.condition.SearchStrategy; +import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.boot.validation.MessageInterpolatorFactory; import org.springframework.boot.validation.beanvalidation.FilteredMethodValidationPostProcessor; import org.springframework.boot.validation.beanvalidation.MethodValidationExcludeFilter; @@ -44,11 +45,13 @@ * * @author Stephane Nicoll * @author Madhura Bhave + * @author Yanming Zhou * @since 1.5.0 */ @AutoConfiguration @ConditionalOnClass(ExecutableValidator.class) @ConditionalOnResource(resources = "classpath:META-INF/services/jakarta.validation.spi.ValidationProvider") +@EnableConfigurationProperties(ValidationProperties.class) @Import(PrimaryDefaultValidatorPostProcessor.class) public class ValidationAutoConfiguration { @@ -68,11 +71,13 @@ public static LocalValidatorFactoryBean defaultValidator(ApplicationContext appl @Bean @ConditionalOnMissingBean(search = SearchStrategy.CURRENT) public static MethodValidationPostProcessor methodValidationPostProcessor(Environment environment, - ObjectProvider validator, ObjectProvider excludeFilters) { + ValidationProperties validationProperties, ObjectProvider validator, + ObjectProvider excludeFilters) { FilteredMethodValidationPostProcessor processor = new FilteredMethodValidationPostProcessor( excludeFilters.orderedStream()); boolean proxyTargetClass = environment.getProperty("spring.aop.proxy-target-class", Boolean.class, true); processor.setProxyTargetClass(proxyTargetClass); + processor.setAdaptConstraintViolations(validationProperties.getMethod().isAdaptConstraintViolations()); processor.setValidatorProvider(validator); return processor; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/validation/ValidationProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/validation/ValidationProperties.java new file mode 100644 index 000000000000..863ffa2f9b83 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/validation/ValidationProperties.java @@ -0,0 +1,64 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.validation; + +import org.springframework.beans.factory.config.BeanDefinition; +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.context.annotation.Role; + +/** + * {@link ConfigurationProperties @ConfigurationProperties} for validation. + * + * @author Yanming Zhou + * @author Andy Wilkinson + * @since 3.5.0 + */ +@Role(BeanDefinition.ROLE_INFRASTRUCTURE) +@ConfigurationProperties("spring.validation") +public class ValidationProperties { + + private Method method = new Method(); + + public Method getMethod() { + return this.method; + } + + public void setMethod(Method method) { + this.method = method; + } + + /** + * Method validation properties. + */ + public static class Method { + + /** + * Whether to adapt ConstraintViolations to MethodValidationResult. + */ + private boolean adaptConstraintViolations; + + public boolean isAdaptConstraintViolations() { + return this.adaptConstraintViolations; + } + + public void setAdaptConstraintViolations(boolean adaptConstraintViolations) { + this.adaptConstraintViolations = adaptConstraintViolations; + } + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/ServerProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/ServerProperties.java index 9be86511b67b..9702d8fe72ba 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/ServerProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/ServerProperties.java @@ -34,7 +34,6 @@ import org.springframework.boot.context.properties.DeprecatedConfigurationProperty; import org.springframework.boot.context.properties.NestedConfigurationProperty; import org.springframework.boot.convert.DurationUnit; -import org.springframework.boot.system.JavaVersion; import org.springframework.boot.web.server.Compression; import org.springframework.boot.web.server.Cookie; import org.springframework.boot.web.server.Http2; @@ -515,18 +514,15 @@ public static class Tomcat { private DataSize maxHttpResponseHeaderSize = DataSize.ofKilobytes(8); /** - * Whether to use APR. If running on Java below 24, the default value is - * 'WHEN_AVAILABLE'. On Java 24 or later, the default value is 'NEVER'. + * Maximum number of parameters (GET plus POST) that will be automatically parsed + * by the container. A value of less than 0 means no limit. */ - private UseApr useApr; + private int maxParameterCount = 10000; - public DataSize getMaxHttpFormPostSize() { - return this.maxHttpFormPostSize; - } - - public void setMaxHttpFormPostSize(DataSize maxHttpFormPostSize) { - this.maxHttpFormPostSize = maxHttpFormPostSize; - } + /** + * Whether to use APR. + */ + private UseApr useApr = UseApr.NEVER; public Accesslog getAccesslog() { return this.accesslog; @@ -676,11 +672,23 @@ public void setMaxHttpResponseHeaderSize(DataSize maxHttpResponseHeaderSize) { this.maxHttpResponseHeaderSize = maxHttpResponseHeaderSize; } + public DataSize getMaxHttpFormPostSize() { + return this.maxHttpFormPostSize; + } + + public void setMaxHttpFormPostSize(DataSize maxHttpFormPostSize) { + this.maxHttpFormPostSize = maxHttpFormPostSize; + } + + public int getMaxParameterCount() { + return this.maxParameterCount; + } + + public void setMaxParameterCount(int maxParameterCount) { + this.maxParameterCount = maxParameterCount; + } + public UseApr getUseApr() { - if (this.useApr == null) { - return JavaVersion.getJavaVersion().isEqualOrNewerThan(JavaVersion.TWENTY_FOUR) ? UseApr.NEVER - : UseApr.WHEN_AVAILABLE; - } return this.useApr; } @@ -1148,7 +1156,7 @@ public enum UseApr { WHEN_AVAILABLE, /** - * Never user APR. + * Never use APR. */ NEVER diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/client/HttpMessageConvertersRestClientCustomizer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/client/HttpMessageConvertersRestClientCustomizer.java index c5372d5d7f84..0ee1099edd6b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/client/HttpMessageConvertersRestClientCustomizer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/client/HttpMessageConvertersRestClientCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ public class HttpMessageConvertersRestClientCustomizer implements RestClientCust private final Iterable> messageConverters; public HttpMessageConvertersRestClientCustomizer(HttpMessageConverter... messageConverters) { - Assert.notNull(messageConverters, "MessageConverters must not be null"); + Assert.notNull(messageConverters, "'messageConverters' must not be null"); this.messageConverters = Arrays.asList(messageConverters); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/embedded/TomcatWebServerFactoryCustomizer.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/embedded/TomcatWebServerFactoryCustomizer.java index 6feadf329bf0..c60ce2cffe02 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/embedded/TomcatWebServerFactoryCustomizer.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/embedded/TomcatWebServerFactoryCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -119,6 +119,8 @@ public void customize(ConfigurableTomcatWebServerFactory factory) { .asInt(DataSize::toBytes) .when((maxHttpFormPostSize) -> maxHttpFormPostSize != 0) .to((maxHttpFormPostSize) -> customizeMaxHttpFormPostSize(factory, maxHttpFormPostSize)); + map.from(properties::getMaxParameterCount) + .to((maxParameterCount) -> customizeMaxParameterCount(factory, maxParameterCount)); map.from(properties::getAccesslog) .when(ServerProperties.Tomcat.Accesslog::isEnabled) .to((enabled) -> customizeAccessLog(factory)); @@ -292,6 +294,10 @@ private void customizeMaxHttpFormPostSize(ConfigurableTomcatWebServerFactory fac factory.addConnectorCustomizers((connector) -> connector.setMaxPostSize(maxHttpFormPostSize)); } + private void customizeMaxParameterCount(ConfigurableTomcatWebServerFactory factory, int maxParameterCount) { + factory.addConnectorCustomizers((connector) -> connector.setMaxParameterCount(maxParameterCount)); + } + private void customizeAccessLog(ConfigurableTomcatWebServerFactory factory) { ServerProperties.Tomcat tomcatProperties = this.serverProperties.getTomcat(); AccessLogValve valve = new AccessLogValve(); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/ReactiveMultipartProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/ReactiveMultipartProperties.java index 01b4e005a2eb..f3565a74ebca 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/ReactiveMultipartProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/ReactiveMultipartProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ * @author Chris Bono * @since 2.6.0 */ -@ConfigurationProperties(prefix = "spring.webflux.multipart") +@ConfigurationProperties("spring.webflux.multipart") public class ReactiveMultipartProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/ReactiveWebServerFactoryAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/ReactiveWebServerFactoryAutoConfiguration.java index 18ba60c405f5..806cadedd0f3 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/ReactiveWebServerFactoryAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/ReactiveWebServerFactoryAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -77,7 +77,7 @@ public TomcatReactiveWebServerFactoryCustomizer tomcatReactiveWebServerFactoryCu @Bean @ConditionalOnMissingBean - @ConditionalOnProperty(value = "server.forward-headers-strategy", havingValue = "framework") + @ConditionalOnProperty(name = "server.forward-headers-strategy", havingValue = "framework") public ForwardedHeaderTransformer forwardedHeaderTransformer() { return new ForwardedHeaderTransformer(); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfiguration.java index 96218e0fe3d9..0ba0a8ef489e 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,9 +28,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.AutoConfigureOrder; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.http.codec.CodecsAutoConfiguration; import org.springframework.boot.autoconfigure.task.TaskExecutionAutoConfiguration; @@ -118,7 +118,7 @@ public class WebFluxAutoConfiguration { @Bean @ConditionalOnMissingBean(HiddenHttpMethodFilter.class) - @ConditionalOnProperty(prefix = "spring.webflux.hiddenmethod.filter", name = "enabled") + @ConditionalOnBooleanProperty("spring.webflux.hiddenmethod.filter.enabled") public OrderedHiddenHttpMethodFilter hiddenHttpMethodFilter() { return new OrderedHiddenHttpMethodFilter(); } @@ -359,7 +359,7 @@ ResourceChainResourceHandlerRegistrationCustomizer resourceHandlerRegistrationCu } @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.webflux.problemdetails", name = "enabled", havingValue = "true") + @ConditionalOnBooleanProperty("spring.webflux.problemdetails.enabled") static class ProblemDetailsErrorHandlingConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxProperties.java index 618b5164f919..dc4e62095530 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * @author Vedran Pavic * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.webflux") +@ConfigurationProperties("spring.webflux") public class WebFluxProperties { /** diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebSessionIdResolverAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebSessionIdResolverAutoConfiguration.java index 797910b8ea37..ae1da0b818f8 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebSessionIdResolverAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebSessionIdResolverAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -77,12 +77,7 @@ private void initializeCookie(ResponseCookieBuilder builder) { map.from(cookie::getSecure).to(builder::secure); map.from(cookie::getMaxAge).to(builder::maxAge); map.from(cookie::getPartitioned).to(builder::partitioned); - map.from(getSameSite(cookie)).to(builder::sameSite); - } - - private String getSameSite(Cookie properties) { - SameSite sameSite = properties.getSameSite(); - return (sameSite != null) ? sameSite.attributeValue() : null; + map.from(cookie::getSameSite).as(SameSite::attributeValue).to(builder::sameSite); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/error/AbstractErrorWebExceptionHandler.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/error/AbstractErrorWebExceptionHandler.java index c601654d0e42..c7315504d59f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/error/AbstractErrorWebExceptionHandler.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/error/AbstractErrorWebExceptionHandler.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -85,9 +85,9 @@ public abstract class AbstractErrorWebExceptionHandler implements ErrorWebExcept */ public AbstractErrorWebExceptionHandler(ErrorAttributes errorAttributes, Resources resources, ApplicationContext applicationContext) { - Assert.notNull(errorAttributes, "ErrorAttributes must not be null"); - Assert.notNull(resources, "Resources must not be null"); - Assert.notNull(applicationContext, "ApplicationContext must not be null"); + Assert.notNull(errorAttributes, "'errorAttributes' must not be null"); + Assert.notNull(resources, "'resources' must not be null"); + Assert.notNull(applicationContext, "'applicationContext' must not be null"); this.errorAttributes = errorAttributes; this.resources = resources; this.applicationContext = applicationContext; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/AutoConfiguredWebClientSsl.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/AutoConfiguredWebClientSsl.java index 503f42a35d56..7460da016b72 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/AutoConfiguredWebClientSsl.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/AutoConfiguredWebClientSsl.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,6 +18,8 @@ import java.util.function.Consumer; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorSettings; import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.ssl.SslBundles; import org.springframework.http.client.reactive.ClientHttpConnector; @@ -30,12 +32,16 @@ */ class AutoConfiguredWebClientSsl implements WebClientSsl { - private final ClientHttpConnectorFactory clientHttpConnectorFactory; + private final ClientHttpConnectorBuilder connectorBuilder; + + private final ClientHttpConnectorSettings settings; private final SslBundles sslBundles; - AutoConfiguredWebClientSsl(ClientHttpConnectorFactory clientHttpConnectorFactory, SslBundles sslBundles) { - this.clientHttpConnectorFactory = clientHttpConnectorFactory; + AutoConfiguredWebClientSsl(ClientHttpConnectorBuilder connectorBuilder, ClientHttpConnectorSettings settings, + SslBundles sslBundles) { + this.connectorBuilder = connectorBuilder; + this.settings = settings; this.sslBundles = sslBundles; } @@ -47,7 +53,8 @@ public Consumer fromBundle(String bundleName) { @Override public Consumer fromBundle(SslBundle bundle) { return (builder) -> { - ClientHttpConnector connector = this.clientHttpConnectorFactory.createClientHttpConnector(bundle); + ClientHttpConnectorSettings settings = this.settings.withSslBundle(bundle); + ClientHttpConnector connector = this.connectorBuilder.build(settings); builder.clientConnector(connector); }; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorAutoConfiguration.java index 9b5aa365d18c..cfbd92417825 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,53 +16,64 @@ package org.springframework.boot.autoconfigure.web.reactive.function.client; +import java.util.List; + +import reactor.netty.http.client.HttpClient; + +import org.springframework.beans.factory.ObjectProvider; import org.springframework.boot.autoconfigure.AutoConfiguration; -import org.springframework.boot.autoconfigure.AutoConfigureAfter; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; -import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.ssl.SslAutoConfiguration; -import org.springframework.boot.web.reactive.function.client.WebClientCustomizer; +import org.springframework.boot.autoconfigure.http.client.reactive.ClientHttpConnectorBuilderCustomizer; +import org.springframework.boot.autoconfigure.reactor.netty.ReactorNettyConfigurations.ReactorResourceFactoryConfiguration; +import org.springframework.boot.http.client.reactive.ReactorClientHttpConnectorBuilder; import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Import; -import org.springframework.context.annotation.Lazy; import org.springframework.core.annotation.Order; -import org.springframework.http.client.reactive.ClientHttpConnector; +import org.springframework.http.client.ReactorResourceFactory; import org.springframework.web.reactive.function.client.WebClient; /** - * {@link EnableAutoConfiguration Auto-configuration} for {@link ClientHttpConnector}. - *

- * It can produce a {@link org.springframework.http.client.reactive.ClientHttpConnector} - * bean and possibly a companion {@code ResourceFactory} bean, depending on the chosen - * HTTP client library. + * Deprecated {@link EnableAutoConfiguration Auto-configuration} for + * {@link ReactorNettyHttpClientMapper}. * * @author Brian Clozel * @author Phillip Webb * @since 2.1.0 + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of + * {@link org.springframework.boot.autoconfigure.http.client.reactive.ClientHttpConnectorAutoConfiguration} + * and to align with the deprecation of {@link ReactorNettyHttpClientMapper} */ @AutoConfiguration @ConditionalOnClass(WebClient.class) -@AutoConfigureAfter(SslAutoConfiguration.class) -@Import({ ClientHttpConnectorFactoryConfiguration.ReactorNetty.class, - ClientHttpConnectorFactoryConfiguration.HttpClient5.class, - ClientHttpConnectorFactoryConfiguration.JdkClient.class }) +@Deprecated(since = "3.5.0", forRemoval = true) public class ClientHttpConnectorAutoConfiguration { - @Bean - @Lazy - @ConditionalOnMissingBean - ClientHttpConnector webClientHttpConnector(ClientHttpConnectorFactory clientHttpConnectorFactory) { - return clientHttpConnectorFactory.createClientHttpConnector(); - } + @Configuration(proxyBeanMethods = false) + @ConditionalOnClass(HttpClient.class) + @Import(ReactorResourceFactoryConfiguration.class) + @SuppressWarnings("removal") + static class ReactorNetty { + + @Bean + @Order(0) + ClientHttpConnectorBuilderCustomizer reactorNettyHttpClientMapperClientHttpConnectorBuilderCustomizer( + ReactorResourceFactory reactorResourceFactory, + ObjectProvider mapperProvider) { + return applyMappers(mapperProvider.orderedStream().toList()); + } + + private ClientHttpConnectorBuilderCustomizer applyMappers( + List mappers) { + return (builder) -> { + for (ReactorNettyHttpClientMapper mapper : mappers) { + builder = builder.withHttpClientCustomizer(mapper::configure); + } + return builder; + }; + } - @Bean - @Lazy - @Order(0) - @ConditionalOnBean(ClientHttpConnector.class) - public WebClientCustomizer webClientHttpConnectorCustomizer(ClientHttpConnector clientHttpConnector) { - return (builder) -> builder.clientConnector(clientHttpConnector); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorFactoryConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorFactoryConfiguration.java deleted file mode 100644 index d3f9aa1b0d8b..000000000000 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorFactoryConfiguration.java +++ /dev/null @@ -1,81 +0,0 @@ -/* - * Copyright 2012-2023 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.autoconfigure.web.reactive.function.client; - -import org.apache.hc.client5.http.impl.async.HttpAsyncClients; -import org.apache.hc.core5.http.nio.AsyncRequestProducer; -import org.apache.hc.core5.reactive.ReactiveResponseConsumer; - -import org.springframework.beans.factory.ObjectProvider; -import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.reactor.netty.ReactorNettyConfigurations; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.context.annotation.Import; -import org.springframework.http.client.ReactorResourceFactory; - -/** - * Configuration classes for WebClient client connectors. - *

- * Those should be {@code @Import} in a regular auto-configuration class to guarantee - * their order of execution. - * - * @author Brian Clozel - */ -class ClientHttpConnectorFactoryConfiguration { - - @Configuration(proxyBeanMethods = false) - @ConditionalOnClass(reactor.netty.http.client.HttpClient.class) - @ConditionalOnMissingBean(ClientHttpConnectorFactory.class) - @Import(ReactorNettyConfigurations.ReactorResourceFactoryConfiguration.class) - static class ReactorNetty { - - @Bean - ReactorClientHttpConnectorFactory reactorClientHttpConnectorFactory( - ReactorResourceFactory reactorResourceFactory, - ObjectProvider mapperProvider) { - return new ReactorClientHttpConnectorFactory(reactorResourceFactory, mapperProvider::orderedStream); - } - - } - - @Configuration(proxyBeanMethods = false) - @ConditionalOnClass({ HttpAsyncClients.class, AsyncRequestProducer.class, ReactiveResponseConsumer.class }) - @ConditionalOnMissingBean(ClientHttpConnectorFactory.class) - static class HttpClient5 { - - @Bean - HttpComponentsClientHttpConnectorFactory httpComponentsClientHttpConnectorFactory() { - return new HttpComponentsClientHttpConnectorFactory(); - } - - } - - @Configuration(proxyBeanMethods = false) - @ConditionalOnClass(java.net.http.HttpClient.class) - @ConditionalOnMissingBean(ClientHttpConnectorFactory.class) - static class JdkClient { - - @Bean - JdkClientHttpConnectorFactory jdkClientHttpConnectorFactory() { - return new JdkClientHttpConnectorFactory(); - } - - } - -} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/HttpComponentsClientHttpConnectorFactory.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/HttpComponentsClientHttpConnectorFactory.java deleted file mode 100644 index cfae0d165904..000000000000 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/HttpComponentsClientHttpConnectorFactory.java +++ /dev/null @@ -1,64 +0,0 @@ -/* - * Copyright 2012-2023 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.autoconfigure.web.reactive.function.client; - -import javax.net.ssl.SSLContext; - -import org.apache.hc.client5.http.impl.async.HttpAsyncClientBuilder; -import org.apache.hc.client5.http.impl.async.HttpAsyncClients; -import org.apache.hc.client5.http.impl.nio.PoolingAsyncClientConnectionManagerBuilder; -import org.apache.hc.client5.http.nio.AsyncClientConnectionManager; -import org.apache.hc.core5.http.nio.ssl.BasicClientTlsStrategy; -import org.apache.hc.core5.reactor.ssl.SSLSessionVerifier; - -import org.springframework.boot.ssl.SslBundle; -import org.springframework.boot.ssl.SslOptions; -import org.springframework.http.client.reactive.HttpComponentsClientHttpConnector; - -/** - * {@link ClientHttpConnectorFactory} for {@link HttpComponentsClientHttpConnector}. - * - * @author Phillip Webb - */ -class HttpComponentsClientHttpConnectorFactory - implements ClientHttpConnectorFactory { - - @Override - public HttpComponentsClientHttpConnector createClientHttpConnector(SslBundle sslBundle) { - HttpAsyncClientBuilder builder = HttpAsyncClients.custom().useSystemProperties(); - if (sslBundle != null) { - SslOptions options = sslBundle.getOptions(); - SSLContext sslContext = sslBundle.createSslContext(); - SSLSessionVerifier sessionVerifier = (endpoint, sslEngine) -> { - if (options.getCiphers() != null) { - sslEngine.setEnabledCipherSuites(options.getCiphers()); - } - if (options.getEnabledProtocols() != null) { - sslEngine.setEnabledProtocols(options.getEnabledProtocols()); - } - return null; - }; - BasicClientTlsStrategy tlsStrategy = new BasicClientTlsStrategy(sslContext, sessionVerifier); - AsyncClientConnectionManager connectionManager = PoolingAsyncClientConnectionManagerBuilder.create() - .setTlsStrategy(tlsStrategy) - .build(); - builder.setConnectionManager(connectionManager); - } - return new HttpComponentsClientHttpConnector(builder.build()); - } - -} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/JdkClientHttpConnectorFactory.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/JdkClientHttpConnectorFactory.java deleted file mode 100644 index 695278f414fb..000000000000 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/JdkClientHttpConnectorFactory.java +++ /dev/null @@ -1,49 +0,0 @@ -/* - * Copyright 2012-2023 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.autoconfigure.web.reactive.function.client; - -import java.net.http.HttpClient; -import java.net.http.HttpClient.Builder; - -import javax.net.ssl.SSLParameters; - -import org.springframework.boot.ssl.SslBundle; -import org.springframework.boot.ssl.SslOptions; -import org.springframework.http.client.reactive.JdkClientHttpConnector; - -/** - * {@link ClientHttpConnectorFactory} for {@link JdkClientHttpConnector}. - * - * @author Phillip Webb - */ -class JdkClientHttpConnectorFactory implements ClientHttpConnectorFactory { - - @Override - public JdkClientHttpConnector createClientHttpConnector(SslBundle sslBundle) { - Builder builder = HttpClient.newBuilder(); - if (sslBundle != null) { - SslOptions options = sslBundle.getOptions(); - builder.sslContext(sslBundle.createSslContext()); - SSLParameters parameters = new SSLParameters(); - parameters.setCipherSuites(options.getCiphers()); - parameters.setProtocols(options.getEnabledProtocols()); - builder.sslParameters(parameters); - } - return new JdkClientHttpConnector(builder.build()); - } - -} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorClientHttpConnectorFactory.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorClientHttpConnectorFactory.java deleted file mode 100644 index 3f596126ac51..000000000000 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorClientHttpConnectorFactory.java +++ /dev/null @@ -1,100 +0,0 @@ -/* - * Copyright 2012-2023 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.autoconfigure.web.reactive.function.client; - -import java.util.ArrayList; -import java.util.List; -import java.util.function.Supplier; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -import javax.net.ssl.SSLException; - -import io.netty.handler.ssl.SslContextBuilder; -import reactor.netty.http.client.HttpClient; -import reactor.netty.tcp.SslProvider.SslContextSpec; - -import org.springframework.boot.ssl.SslBundle; -import org.springframework.boot.ssl.SslManagerBundle; -import org.springframework.boot.ssl.SslOptions; -import org.springframework.http.client.ReactorResourceFactory; -import org.springframework.http.client.reactive.ReactorClientHttpConnector; -import org.springframework.util.function.ThrowingConsumer; - -/** - * {@link ClientHttpConnectorFactory} for {@link ReactorClientHttpConnector}. - * - * @author Phillip Webb - * @author Fernando Cappi - */ -class ReactorClientHttpConnectorFactory implements ClientHttpConnectorFactory { - - private final ReactorResourceFactory reactorResourceFactory; - - private final Supplier> mappers; - - ReactorClientHttpConnectorFactory(ReactorResourceFactory reactorResourceFactory) { - this(reactorResourceFactory, Stream::empty); - } - - ReactorClientHttpConnectorFactory(ReactorResourceFactory reactorResourceFactory, - Supplier> mappers) { - this.reactorResourceFactory = reactorResourceFactory; - this.mappers = mappers; - } - - @Override - public ReactorClientHttpConnector createClientHttpConnector(SslBundle sslBundle) { - List mappers = this.mappers.get() - .collect(Collectors.toCollection(ArrayList::new)); - if (sslBundle != null) { - mappers.add(new SslConfigurer(sslBundle)); - } - return new ReactorClientHttpConnector(this.reactorResourceFactory, - ReactorNettyHttpClientMapper.of(mappers)::configure); - } - - /** - * Configures the Netty {@link HttpClient} with SSL. - */ - private static class SslConfigurer implements ReactorNettyHttpClientMapper { - - private final SslBundle sslBundle; - - SslConfigurer(SslBundle sslBundle) { - this.sslBundle = sslBundle; - } - - @Override - public HttpClient configure(HttpClient httpClient) { - return httpClient.secure(ThrowingConsumer.of(this::customizeSsl).throwing(IllegalStateException::new)); - } - - private void customizeSsl(SslContextSpec spec) throws SSLException { - SslOptions options = this.sslBundle.getOptions(); - SslManagerBundle managers = this.sslBundle.getManagers(); - SslContextBuilder builder = SslContextBuilder.forClient() - .keyManager(managers.getKeyManagerFactory()) - .trustManager(managers.getTrustManagerFactory()) - .ciphers(SslOptions.asSet(options.getCiphers())) - .protocols(options.getEnabledProtocols()); - spec.sslContext(builder.build()); - } - - } - -} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorNettyHttpClientMapper.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorNettyHttpClientMapper.java index 2589b002d51a..4f302c3d12eb 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorNettyHttpClientMapper.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorNettyHttpClientMapper.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,6 +20,8 @@ import reactor.netty.http.client.HttpClient; +import org.springframework.boot.autoconfigure.http.client.reactive.ClientHttpConnectorBuilderCustomizer; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder; import org.springframework.http.client.reactive.ReactorClientHttpConnector; import org.springframework.util.Assert; @@ -30,8 +32,12 @@ * @author Brian Clozel * @author Phillip Webb * @since 2.3.0 + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of + * {@link ClientHttpConnectorBuilderCustomizer} or declaring a pre-configured + * {@link ClientHttpConnectorBuilder} bean */ @FunctionalInterface +@Deprecated(since = "3.5.0", forRemoval = true) public interface ReactorNettyHttpClientMapper { /** @@ -48,7 +54,7 @@ public interface ReactorNettyHttpClientMapper { * @since 3.1.1 */ static ReactorNettyHttpClientMapper of(Collection mappers) { - Assert.notNull(mappers, "Mappers must not be null"); + Assert.notNull(mappers, "'mappers' must not be null"); return of(mappers.toArray(ReactorNettyHttpClientMapper[]::new)); } @@ -59,7 +65,7 @@ static ReactorNettyHttpClientMapper of(Collection * @since 3.1.1 */ static ReactorNettyHttpClientMapper of(ReactorNettyHttpClientMapper... mappers) { - Assert.notNull(mappers, "Mappers must not be null"); + Assert.notNull(mappers, "'mappers' must not be null"); return (httpClient) -> { for (ReactorNettyHttpClientMapper mapper : mappers) { httpClient = mapper.configure(httpClient); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/WebClientAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/WebClientAutoConfiguration.java index b0d86b145519..6593c50f78e2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/WebClientAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/function/client/WebClientAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,14 +23,19 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.boot.autoconfigure.http.client.reactive.ClientHttpConnectorAutoConfiguration; import org.springframework.boot.autoconfigure.http.codec.CodecsAutoConfiguration; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorSettings; import org.springframework.boot.ssl.SslBundles; import org.springframework.boot.web.codec.CodecCustomizer; import org.springframework.boot.web.reactive.function.client.WebClientCustomizer; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Lazy; import org.springframework.context.annotation.Scope; import org.springframework.core.annotation.Order; +import org.springframework.http.client.reactive.ClientHttpConnector; import org.springframework.web.reactive.function.client.WebClient; /** @@ -42,6 +47,7 @@ * will receive a newly cloned instance of the builder. * * @author Brian Clozel + * @author Phillip Webb * @since 2.0.0 */ @AutoConfiguration(after = { CodecsAutoConfiguration.class, ClientHttpConnectorAutoConfiguration.class }) @@ -57,12 +63,20 @@ public WebClient.Builder webClientBuilder(ObjectProvider cu return builder; } + @Bean + @Lazy + @Order(0) + @ConditionalOnBean(ClientHttpConnector.class) + public WebClientCustomizer webClientHttpConnectorCustomizer(ClientHttpConnector clientHttpConnector) { + return (builder) -> builder.clientConnector(clientHttpConnector); + } + @Bean @ConditionalOnMissingBean(WebClientSsl.class) @ConditionalOnBean(SslBundles.class) - AutoConfiguredWebClientSsl webClientSsl(ClientHttpConnectorFactory clientHttpConnectorFactory, - SslBundles sslBundles) { - return new AutoConfiguredWebClientSsl(clientHttpConnectorFactory, sslBundles); + AutoConfiguredWebClientSsl webClientSsl(ClientHttpConnectorBuilder clientHttpConnectorBuilder, + ClientHttpConnectorSettings clientHttpConnectorSettings, SslBundles sslBundles) { + return new AutoConfiguredWebClientSsl(clientHttpConnectorBuilder, clientHttpConnectorSettings, sslBundles); } @Configuration(proxyBeanMethods = false) diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/DispatcherServletRegistrationBean.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/DispatcherServletRegistrationBean.java index 9c6625cfe996..3eccf8887cfb 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/DispatcherServletRegistrationBean.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/DispatcherServletRegistrationBean.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,7 +42,7 @@ public class DispatcherServletRegistrationBean extends ServletRegistrationBean mediaTypes = this.mvcProperties.getContentnegotiation().getMediaTypes(); + Map mediaTypes = contentnegotiation.getMediaTypes(); mediaTypes.forEach(configurer::mediaType); + List defaultContentTypes = contentnegotiation.getDefaultContentTypes(); + if (!CollectionUtils.isEmpty(defaultContentTypes)) { + configurer.defaultContentType(defaultContentTypes.toArray(new MediaType[0])); + } } @Bean @@ -666,7 +671,7 @@ private ResourceResolver getVersionResourceResolver(Strategy properties) { } @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.mvc.problemdetails", name = "enabled", havingValue = "true") + @ConditionalOnBooleanProperty("spring.mvc.problemdetails.enabled") static class ProblemDetailsErrorHandlingConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcProperties.java index 53d6bbfdd1af..8de413f29d76 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,7 +17,9 @@ package org.springframework.boot.autoconfigure.web.servlet; import java.time.Duration; +import java.util.ArrayList; import java.util.LinkedHashMap; +import java.util.List; import java.util.Map; import org.springframework.boot.context.properties.ConfigurationProperties; @@ -36,7 +38,7 @@ * @author Vedran Pavic * @since 2.0.0 */ -@ConfigurationProperties(prefix = "spring.mvc") +@ConfigurationProperties("spring.mvc") public class WebMvcProperties { /** @@ -223,8 +225,8 @@ public String getPath() { } public void setPath(String path) { - Assert.notNull(path, "Path must not be null"); - Assert.isTrue(!path.contains("*"), "Path must not contain wildcards"); + Assert.notNull(path, "'path' must not be null"); + Assert.isTrue(!path.contains("*"), "'path' must not contain wildcards"); this.path = path; } @@ -306,6 +308,11 @@ public static class Contentnegotiation { */ private boolean favorParameter = false; + /** + * Query parameter name to use when "favor-parameter" is enabled. + */ + private String parameterName; + /** * Map file extensions to media types for content negotiation. For instance, yml * to text/yaml. @@ -313,9 +320,10 @@ public static class Contentnegotiation { private Map mediaTypes = new LinkedHashMap<>(); /** - * Query parameter name to use when "favor-parameter" is enabled. + * List of default content types to be used when no specific content type is + * requested. */ - private String parameterName; + private List defaultContentTypes = new ArrayList<>(); public boolean isFavorParameter() { return this.favorParameter; @@ -325,6 +333,14 @@ public void setFavorParameter(boolean favorParameter) { this.favorParameter = favorParameter; } + public String getParameterName() { + return this.parameterName; + } + + public void setParameterName(String parameterName) { + this.parameterName = parameterName; + } + public Map getMediaTypes() { return this.mediaTypes; } @@ -333,12 +349,12 @@ public void setMediaTypes(Map mediaTypes) { this.mediaTypes = mediaTypes; } - public String getParameterName() { - return this.parameterName; + public List getDefaultContentTypes() { + return this.defaultContentTypes; } - public void setParameterName(String parameterName) { - this.parameterName = parameterName; + public void setDefaultContentTypes(List defaultContentTypes) { + this.defaultContentTypes = defaultContentTypes; } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/AbstractErrorController.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/AbstractErrorController.java index 3fbfbd3b38c1..857771055550 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/AbstractErrorController.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/AbstractErrorController.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,7 +56,7 @@ public AbstractErrorController(ErrorAttributes errorAttributes) { } public AbstractErrorController(ErrorAttributes errorAttributes, List errorViewResolvers) { - Assert.notNull(errorAttributes, "ErrorAttributes must not be null"); + Assert.notNull(errorAttributes, "'errorAttributes' must not be null"); this.errorAttributes = errorAttributes; this.errorViewResolvers = sortErrorViewResolvers(errorViewResolvers); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/BasicErrorController.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/BasicErrorController.java index add6851f2ec0..4a082f2ec35a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/BasicErrorController.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/BasicErrorController.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -78,7 +78,7 @@ public BasicErrorController(ErrorAttributes errorAttributes, ErrorProperties err public BasicErrorController(ErrorAttributes errorAttributes, ErrorProperties errorProperties, List errorViewResolvers) { super(errorAttributes, errorViewResolvers); - Assert.notNull(errorProperties, "ErrorProperties must not be null"); + Assert.notNull(errorProperties, "'errorProperties' must not be null"); this.errorProperties = errorProperties; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/DefaultErrorViewResolver.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/DefaultErrorViewResolver.java index 23474a1dc5b7..6d3415c9bdca 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/DefaultErrorViewResolver.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/DefaultErrorViewResolver.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -81,8 +81,8 @@ public class DefaultErrorViewResolver implements ErrorViewResolver, Ordered { * @since 2.4.0 */ public DefaultErrorViewResolver(ApplicationContext applicationContext, Resources resources) { - Assert.notNull(applicationContext, "ApplicationContext must not be null"); - Assert.notNull(resources, "Resources must not be null"); + Assert.notNull(applicationContext, "'applicationContext' must not be null"); + Assert.notNull(resources, "'resources' must not be null"); this.applicationContext = applicationContext; this.resources = resources; this.templateAvailabilityProviders = new TemplateAvailabilityProviders(applicationContext); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/ErrorMvcAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/ErrorMvcAutoConfiguration.java index fac3b91d7f8b..9252c7fa40d2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/ErrorMvcAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/error/ErrorMvcAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,9 +35,9 @@ import org.springframework.boot.autoconfigure.condition.ConditionMessage; import org.springframework.boot.autoconfigure.condition.ConditionOutcome; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication.Type; import org.springframework.boot.autoconfigure.condition.SearchStrategy; @@ -142,7 +142,7 @@ DefaultErrorViewResolver conventionErrorViewResolver() { } @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "server.error.whitelabel", name = "enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "server.error.whitelabel.enabled", matchIfMissing = true) @Conditional(ErrorTemplateMissingCondition.class) protected static class WhitelabelErrorViewConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/webservices/WebServicesProperties.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/webservices/WebServicesProperties.java index 9f1014b4c7f2..649c4fd80978 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/webservices/WebServicesProperties.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/webservices/WebServicesProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Stephane Nicoll * @since 1.4.0 */ -@ConfigurationProperties(prefix = "spring.webservices") +@ConfigurationProperties("spring.webservices") public class WebServicesProperties { /** @@ -44,9 +44,9 @@ public String getPath() { } public void setPath(String path) { - Assert.notNull(path, "Path must not be null"); - Assert.isTrue(path.length() > 1, "Path must have length greater than 1"); - Assert.isTrue(path.startsWith("/"), "Path must start with '/'"); + Assert.notNull(path, "'path' must not be null"); + Assert.isTrue(path.length() > 1, "'path' must have length greater than 1"); + Assert.isTrue(path.startsWith("/"), "'path' must start with '/'"); this.path = path; } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/additional-spring-configuration-metadata.json b/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/additional-spring-configuration-metadata.json index 44ba87e00956..b5e8c3c24a55 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/additional-spring-configuration-metadata.json +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/additional-spring-configuration-metadata.json @@ -269,7 +269,7 @@ }, { "name": "server.ssl.bundle", - "description": "The name of a configured SSL bundle." + "description": "Name of a configured SSL bundle." }, { "name": "server.ssl.certificate", @@ -1511,6 +1511,99 @@ "name": "spring.graphql.schema.locations", "defaultValue": "classpath:graphql/**/" }, + { + "name": "spring.groovy.template.configuration.auto-escape", + "deprecation": { + "replacement": "spring.groovy.template.auto-escape", + "level": "warning" + } + }, + { + "name": "spring.groovy.template.configuration.auto-indent", + "deprecation": { + "replacement": "spring.groovy.template.auto-indent", + "level": "warning" + } + }, + { + "name": "spring.groovy.template.configuration.auto-indent-string", + "deprecation": { + "replacement": "spring.groovy.template.auto-indent-string", + "level": "warning" + } + }, + { + "name": "spring.groovy.template.configuration.auto-new-line", + "deprecation": { + "replacement": "spring.groovy.template.auto-new-line", + "level": "warning", + "since": "3.5.0" + } + }, + { + "name": "spring.groovy.template.configuration.base-template-class", + "deprecation": { + "replacement": "spring.groovy.template.base-template-class", + "level": "warning", + "since": "3.5.0" + } + }, + { + "name": "spring.groovy.template.configuration.cache-templates", + "deprecation": { + "replacement": "spring.groovy.template.cache", + "level": "warning", + "since": "3.5.0" + } + }, + { + "name": "spring.groovy.template.configuration.declaration-encoding", + "deprecation": { + "replacement": "spring.groovy.template.declaration-encoding", + "level": "warning", + "since": "3.5.0" + } + }, + { + "name": "spring.groovy.template.configuration.expand-empty-elements", + "deprecation": { + "replacement": "spring.groovy.template.expand-empty-elements", + "level": "warning", + "since": "3.5.0" + } + }, + { + "name": "spring.groovy.template.configuration.locale", + "deprecation": { + "replacement": "spring.groovy.template.locale", + "level": "warning", + "since": "3.5.0" + } + }, + { + "name": "spring.groovy.template.configuration.new-line-string", + "deprecation": { + "replacement": "spring.groovy.template.new-line-string", + "level": "warning", + "since": "3.5.0" + } + }, + { + "name": "spring.groovy.template.configuration.resource-loader-path", + "deprecation": { + "replacement": "spring.groovy.template.resource-loader-path", + "level": "warning", + "since": "3.5.0" + } + }, + { + "name": "spring.groovy.template.configuration.use-double-quotes", + "deprecation": { + "replacement": "spring.groovy.template.use-double-quotes", + "level": "warning", + "since": "3.5.0" + } + }, { "name": "spring.groovy.template.prefix", "defaultValue": "" @@ -1522,11 +1615,8 @@ { "name": "spring.http.converters.preferred-json-mapper", "type": "java.lang.String", - "description": "Preferred JSON mapper to use for HTTP message conversion. By default, auto-detected according to the environment.", - "deprecation": { - "replacement": "spring.mvc.converters.preferred-json-mapper", - "level": "error" - } + "defaultValue": "jackson", + "description": "Preferred JSON mapper to use for HTTP message conversion. By default, auto-detected according to the environment. Supported values are 'jackson', 'gson', and 'jsonb'. When other json mapping libraries (such as kotlinx.serialization) are present, use a custom HttpMessageConverters bean to control the preferred mapper." }, { "name": "spring.http.encoding.charset", @@ -2014,7 +2104,11 @@ "name": "spring.mvc.converters.preferred-json-mapper", "type": "java.lang.String", "defaultValue": "jackson", - "description": "Preferred JSON mapper to use for HTTP message conversion. By default, auto-detected according to the environment. Supported values are 'jackson', 'gson', and 'jsonb'. When other json mapping libraries (such as kotlinx.serialization) are present, use a custom HttpMessageConverters bean to control the preferred mapper." + "description": "Preferred JSON mapper to use for HTTP message conversion. By default, auto-detected according to the environment. Supported values are 'jackson', 'gson', and 'jsonb'. When other json mapping libraries (such as kotlinx.serialization) are present, use a custom HttpMessageConverters bean to control the preferred mapper.", + "deprecation": { + "replacement": "spring.http.converters.preferred-json-mapper", + "level": "error" + } }, { "name": "spring.mvc.date-format", @@ -2606,7 +2700,7 @@ }, { "name": "spring.rsocket.server.ssl.bundle", - "description": "The name of a configured SSL bundle." + "description": "Name of a configured SSL bundle." }, { "name": "spring.rsocket.server.ssl.certificate", @@ -2924,6 +3018,68 @@ } ] }, + { + "name": "spring.data.mongodb.protocol", + "values": [ + { + "value": "mongodb" + }, + { + "value": "mongodb+srv" + } + ], + "providers": [ + { + "name": "any" + } + ] + }, + { + "name": "spring.data.redis.lettuce.read-from", + "values": [ + { + "value": "any", + "description": "Read from any node." + }, + { + "value": "any-replica", + "description": "Read from any replica node." + }, + { + "value": "lowest-latency", + "description": "Read from the node with the lowest latency during topology discovery." + }, + { + "value": "regex:", + "description": "Read from any node that has RedisURI matching with the given pattern." + }, + { + "value": "replica", + "description": "Read from the replica only." + }, + { + "value": "replica-preferred", + "description": "Read preferred from replica and fall back to upstream if no replica is available." + }, + { + "value": "subnet:", + "description": "Read from any node in the subnets." + }, + { + "value": "upstream", + "description": "Read from the upstream only." + }, + { + "value": "upstream-preferred", + "description": "Read preferred from the upstream and fall back to a replica if the upstream is not available." + } + ], + "providers": [ + { + "name": "any" + } + ] + }, { "name": "spring.datasource.data", "providers": [ @@ -3018,6 +3174,25 @@ } ] }, + { + "name": "spring.http.converters.preferred-json-mapper", + "values": [ + { + "value": "gson" + }, + { + "value": "jackson" + }, + { + "value": "jsonb" + } + ], + "providers": [ + { + "name": "any" + } + ] + }, { "name": "spring.jms.listener.session.acknowledge-mode", "values": [ @@ -3356,5 +3531,48 @@ } ] } - ] + ], + "ignored": { + "properties": [ + { + "name": "spring.datasource.dbcp2.driver" + }, + { + "name": "spring.datasource.hikari.credentials" + }, + { + "name": "spring.datasource.hikari.data-source-properties" + }, + { + "name": "spring.datasource.hikari.exception-override" + }, + { + "name": "spring.datasource.hikari.health-check-properties" + }, + { + "name": "spring.datasource.hikari.metrics-tracker-factory" + }, + { + "name": "spring.datasource.hikari.scheduled-executor" + }, + { + "name": "spring.datasource.oracleucp.connection-factory-properties" + }, + { + "name": "spring.datasource.oracleucp.connection-properties" + }, + { + "name": "spring.datasource.oracleucp.connection-wait-duration-in-millis" + }, + { + "name": "spring.datasource.oracleucp.hostname-resolver" + }, + { + "name": "spring.datasource.oracleucp.pdb-roles" + }, + { + "name": "spring.datasource.tomcat.db-properties" + } + ] + } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring.factories b/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring.factories index 450bee2bc69d..23dc687cea13 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring.factories +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring.factories @@ -27,6 +27,7 @@ org.springframework.boot.autoconfigure.data.redis.RedisUrlSyntaxFailureAnalyzer, org.springframework.boot.autoconfigure.diagnostics.analyzer.NoSuchBeanDefinitionFailureAnalyzer,\ org.springframework.boot.autoconfigure.jdbc.DataSourceBeanCreationFailureAnalyzer,\ org.springframework.boot.autoconfigure.jdbc.HikariDriverConfigurationFailureAnalyzer,\ +org.springframework.boot.autoconfigure.jooq.JaxbNotAvailableExceptionFailureAnalyzer,\ org.springframework.boot.autoconfigure.jooq.NoDslContextBeanFailureAnalyzer,\ org.springframework.boot.autoconfigure.r2dbc.ConnectionFactoryBeanCreationFailureAnalyzer,\ org.springframework.boot.autoconfigure.r2dbc.MissingR2dbcPoolDependencyFailureAnalyzer,\ diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports b/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports index 05740b676a3c..8325b51ba04b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports @@ -63,6 +63,7 @@ org.springframework.boot.autoconfigure.hazelcast.HazelcastAutoConfiguration org.springframework.boot.autoconfigure.hazelcast.HazelcastJpaDependencyAutoConfiguration org.springframework.boot.autoconfigure.http.HttpMessageConvertersAutoConfiguration org.springframework.boot.autoconfigure.http.client.HttpClientAutoConfiguration +org.springframework.boot.autoconfigure.http.client.reactive.ClientHttpConnectorAutoConfiguration org.springframework.boot.autoconfigure.http.codec.CodecsAutoConfiguration org.springframework.boot.autoconfigure.info.ProjectInfoAutoConfiguration org.springframework.boot.autoconfigure.integration.IntegrationAutoConfiguration diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/amqp/PropertiesRabbitConnectionDetailsTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/amqp/PropertiesRabbitConnectionDetailsTests.java index a3b0162b0667..2db99c686aca 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/amqp/PropertiesRabbitConnectionDetailsTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/amqp/PropertiesRabbitConnectionDetailsTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,11 +18,15 @@ import java.util.List; +import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; import org.springframework.boot.autoconfigure.amqp.RabbitConnectionDetails.Address; +import org.springframework.boot.ssl.DefaultSslBundleRegistry; +import org.springframework.boot.ssl.SslBundle; import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.Mockito.mock; /** * Tests for {@link PropertiesRabbitConnectionDetails}. @@ -33,13 +37,24 @@ class PropertiesRabbitConnectionDetailsTests { private static final int DEFAULT_PORT = 5672; + private DefaultSslBundleRegistry sslBundleRegistry; + + private RabbitProperties properties; + + private PropertiesRabbitConnectionDetails propertiesRabbitConnectionDetails; + + @BeforeEach + void setUp() { + this.properties = new RabbitProperties(); + this.sslBundleRegistry = new DefaultSslBundleRegistry(); + this.propertiesRabbitConnectionDetails = new PropertiesRabbitConnectionDetails(this.properties, + this.sslBundleRegistry); + } + @Test void getAddresses() { - RabbitProperties properties = new RabbitProperties(); - properties.setAddresses(List.of("localhost", "localhost:1234", "[::1]", "[::1]:32863")); - PropertiesRabbitConnectionDetails propertiesRabbitConnectionDetails = new PropertiesRabbitConnectionDetails( - properties); - List

addresses = propertiesRabbitConnectionDetails.getAddresses(); + this.properties.setAddresses(List.of("localhost", "localhost:1234", "[::1]", "[::1]:32863")); + List
addresses = this.propertiesRabbitConnectionDetails.getAddresses(); assertThat(addresses.size()).isEqualTo(4); assertThat(addresses.get(0).host()).isEqualTo("localhost"); assertThat(addresses.get(0).port()).isEqualTo(DEFAULT_PORT); @@ -51,4 +66,27 @@ void getAddresses() { assertThat(addresses.get(3).port()).isEqualTo(32863); } + @Test + void shouldReturnSslBundle() { + SslBundle bundle1 = mock(SslBundle.class); + this.sslBundleRegistry.registerBundle("bundle-1", bundle1); + this.properties.getSsl().setBundle("bundle-1"); + SslBundle sslBundle = this.propertiesRabbitConnectionDetails.getSslBundle(); + assertThat(sslBundle).isSameAs(bundle1); + } + + @Test + void shouldReturnNullIfSslIsEnabledButBundleNotSet() { + this.properties.getSsl().setEnabled(true); + SslBundle sslBundle = this.propertiesRabbitConnectionDetails.getSslBundle(); + assertThat(sslBundle).isNull(); + } + + @Test + void shouldReturnNullIfSslIsNotEnabled() { + this.properties.getSsl().setEnabled(false); + SslBundle sslBundle = this.propertiesRabbitConnectionDetails.getSslBundle(); + assertThat(sslBundle).isNull(); + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfigurationTests.java index 5b6a4c3ccecc..c754cdac725e 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfigurationTests.java @@ -39,6 +39,9 @@ import org.springframework.batch.core.configuration.JobRegistry; import org.springframework.batch.core.configuration.annotation.EnableBatchProcessing; import org.springframework.batch.core.configuration.support.DefaultBatchConfiguration; +import org.springframework.batch.core.converter.DefaultJobParametersConverter; +import org.springframework.batch.core.converter.JobParametersConverter; +import org.springframework.batch.core.converter.JsonJobParametersConverter; import org.springframework.batch.core.explore.JobExplorer; import org.springframework.batch.core.job.AbstractJob; import org.springframework.batch.core.launch.JobLauncher; @@ -56,13 +59,13 @@ import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.TestAutoConfigurationPackage; import org.springframework.boot.autoconfigure.batch.BatchAutoConfiguration.SpringBootBatchConfiguration; +import org.springframework.boot.autoconfigure.batch.domain.City; import org.springframework.boot.autoconfigure.flyway.FlywayAutoConfiguration; import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; import org.springframework.boot.autoconfigure.jdbc.DataSourceTransactionManagerAutoConfiguration; import org.springframework.boot.autoconfigure.jdbc.EmbeddedDataSourceConfiguration; import org.springframework.boot.autoconfigure.liquibase.LiquibaseAutoConfiguration; import org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration; -import org.springframework.boot.autoconfigure.orm.jpa.test.City; import org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration; import org.springframework.boot.autoconfigure.transaction.TransactionManagerCustomizationAutoConfiguration; import org.springframework.boot.jdbc.DataSourceBuilder; @@ -90,10 +93,11 @@ import org.springframework.jdbc.datasource.init.DatabasePopulator; import org.springframework.orm.jpa.JpaTransactionManager; import org.springframework.transaction.PlatformTransactionManager; +import org.springframework.transaction.annotation.Isolation; import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatExceptionOfType; -import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; import static org.mockito.BDDMockito.given; import static org.mockito.Mockito.mock; @@ -107,6 +111,7 @@ * @author Mahmoud Ben Hassine * @author Lars Uffmann * @author Lasse Wulff + * @author Yanming Zhou */ @ExtendWith(OutputCaptureExtension.class) class BatchAutoConfigurationTests { @@ -117,23 +122,22 @@ class BatchAutoConfigurationTests { @Test void testDefaultContext() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) - .run((context) -> { - assertThat(context).hasSingleBean(JobRepository.class); - assertThat(context).hasSingleBean(JobLauncher.class); - assertThat(context).hasSingleBean(JobExplorer.class); - assertThat(context).hasSingleBean(JobRegistry.class); - assertThat(context).hasSingleBean(JobOperator.class); - assertThat(context.getBean(BatchProperties.class).getJdbc().getInitializeSchema()) - .isEqualTo(DatabaseInitializationMode.EMBEDDED); - assertThat(new JdbcTemplate(context.getBean(DataSource.class)) - .queryForList("select * from BATCH_JOB_EXECUTION")).isEmpty(); - }); + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class).run((context) -> { + assertThat(context).hasSingleBean(JobRepository.class); + assertThat(context).hasSingleBean(JobLauncher.class); + assertThat(context).hasSingleBean(JobExplorer.class); + assertThat(context).hasSingleBean(JobRegistry.class); + assertThat(context).hasSingleBean(JobOperator.class); + assertThat(context.getBean(BatchProperties.class).getJdbc().getInitializeSchema()) + .isEqualTo(DatabaseInitializationMode.EMBEDDED); + assertThat(new JdbcTemplate(context.getBean(DataSource.class)) + .queryForList("select * from BATCH_JOB_EXECUTION")).isEmpty(); + }); } @Test void autoconfigurationBacksOffEntirelyIfSpringJdbcAbsent() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class) .withClassLoader(new FilteredClassLoader(DatabasePopulator.class)) .run((context) -> { assertThat(context).doesNotHaveBean(JobLauncherApplicationRunner.class); @@ -257,7 +261,7 @@ void testDisableLaunchesJob() { @Test void testDisableSchemaLoader() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class) .withPropertyValues("spring.datasource.generate-unique-name=true", "spring.batch.jdbc.initialize-schema:never") .run((context) -> { @@ -274,7 +278,7 @@ void testDisableSchemaLoader() { @Test void testUsingJpa() { this.contextRunner - .withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class, + .withUserConfiguration(TestJpaConfiguration.class, EmbeddedDataSourceConfiguration.class, HibernateJpaAutoConfiguration.class) .run((context) -> { PlatformTransactionManager transactionManager = context.getBean(PlatformTransactionManager.class); @@ -292,11 +296,9 @@ void testUsingJpa() { @Test @WithPackageResources("custom-schema.sql") void testRenamePrefix() { - this.contextRunner - .withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class, - HibernateJpaAutoConfiguration.class) + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class) .withPropertyValues("spring.datasource.generate-unique-name=true", - "spring.batch.jdbc.schema:classpath:custom-schema.sql", "spring.batch.jdbc.tablePrefix:PREFIX_") + "spring.batch.jdbc.schema:classpath:custom-schema.sql", "spring.batch.jdbc.table-prefix:PREFIX_") .run((context) -> { assertThat(context).hasSingleBean(JobLauncher.class); assertThat(context.getBean(BatchProperties.class).getJdbc().getInitializeSchema()) @@ -313,7 +315,7 @@ void testRenamePrefix() { @Test void testCustomizeJpaTransactionManagerUsingProperties() { this.contextRunner - .withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class, + .withUserConfiguration(TestJpaConfiguration.class, EmbeddedDataSourceConfiguration.class, HibernateJpaAutoConfiguration.class) .withPropertyValues("spring.transaction.default-timeout:30", "spring.transaction.rollback-on-commit-failure:true") @@ -328,7 +330,7 @@ void testCustomizeJpaTransactionManagerUsingProperties() { @Test void testCustomizeDataSourceTransactionManagerUsingProperties() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class) .withPropertyValues("spring.transaction.default-timeout:30", "spring.transaction.rollback-on-commit-failure:true") .run((context) -> { @@ -342,36 +344,32 @@ void testCustomizeDataSourceTransactionManagerUsingProperties() { @Test void testBatchDataSource() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, BatchDataSourceConfiguration.class) - .run((context) -> { - assertThat(context).hasSingleBean(SpringBootBatchConfiguration.class) - .hasSingleBean(BatchDataSourceScriptDatabaseInitializer.class) - .hasBean("batchDataSource"); - DataSource batchDataSource = context.getBean("batchDataSource", DataSource.class); - assertThat(context.getBean(SpringBootBatchConfiguration.class).getDataSource()) - .isEqualTo(batchDataSource); - assertThat(context.getBean(BatchDataSourceScriptDatabaseInitializer.class)) - .hasFieldOrPropertyWithValue("dataSource", batchDataSource); - }); + this.contextRunner.withUserConfiguration(BatchDataSourceConfiguration.class).run((context) -> { + assertThat(context).hasSingleBean(SpringBootBatchConfiguration.class) + .hasSingleBean(BatchDataSourceScriptDatabaseInitializer.class) + .hasBean("batchDataSource"); + DataSource batchDataSource = context.getBean("batchDataSource", DataSource.class); + assertThat(context.getBean(SpringBootBatchConfiguration.class).getDataSource()).isEqualTo(batchDataSource); + assertThat(context.getBean(BatchDataSourceScriptDatabaseInitializer.class)) + .hasFieldOrPropertyWithValue("dataSource", batchDataSource); + }); } @Test void testBatchTransactionManager() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, BatchTransactionManagerConfiguration.class) - .run((context) -> { - assertThat(context).hasSingleBean(SpringBootBatchConfiguration.class); - PlatformTransactionManager batchTransactionManager = context.getBean("batchTransactionManager", - PlatformTransactionManager.class); - assertThat(context.getBean(SpringBootBatchConfiguration.class).getTransactionManager()) - .isEqualTo(batchTransactionManager); - }); + this.contextRunner.withUserConfiguration(BatchTransactionManagerConfiguration.class).run((context) -> { + assertThat(context).hasSingleBean(SpringBootBatchConfiguration.class); + PlatformTransactionManager batchTransactionManager = context.getBean("batchTransactionManager", + PlatformTransactionManager.class); + assertThat(context.getBean(SpringBootBatchConfiguration.class).getTransactionManager()) + .isEqualTo(batchTransactionManager); + }); } @Test void testBatchTaskExecutor() { this.contextRunner - .withUserConfiguration(TestConfiguration.class, BatchTaskExecutorConfiguration.class, - EmbeddedDataSourceConfiguration.class) + .withUserConfiguration(BatchTaskExecutorConfiguration.class, EmbeddedDataSourceConfiguration.class) .run((context) -> { assertThat(context).hasSingleBean(SpringBootBatchConfiguration.class).hasBean("batchTaskExecutor"); TaskExecutor batchTaskExecutor = context.getBean("batchTaskExecutor", TaskExecutor.class); @@ -385,22 +383,20 @@ void testBatchTaskExecutor() { @Test void jobRepositoryBeansDependOnBatchDataSourceInitializer() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) - .run((context) -> { - ConfigurableListableBeanFactory beanFactory = context.getBeanFactory(); - String[] jobRepositoryNames = beanFactory.getBeanNamesForType(JobRepository.class); - assertThat(jobRepositoryNames).isNotEmpty(); - for (String jobRepositoryName : jobRepositoryNames) { - assertThat(beanFactory.getBeanDefinition(jobRepositoryName).getDependsOn()) - .contains("batchDataSourceInitializer"); - } - }); + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class).run((context) -> { + ConfigurableListableBeanFactory beanFactory = context.getBeanFactory(); + String[] jobRepositoryNames = beanFactory.getBeanNamesForType(JobRepository.class); + assertThat(jobRepositoryNames).isNotEmpty(); + for (String jobRepositoryName : jobRepositoryNames) { + assertThat(beanFactory.getBeanDefinition(jobRepositoryName).getDependsOn()) + .contains("batchDataSourceInitializer"); + } + }); } @Test void jobRepositoryBeansDependOnFlyway() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) - .withUserConfiguration(FlywayAutoConfiguration.class) + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class, FlywayAutoConfiguration.class) .withPropertyValues("spring.batch.initialize-schema=never") .run((context) -> { ConfigurableListableBeanFactory beanFactory = context.getBeanFactory(); @@ -416,8 +412,8 @@ void jobRepositoryBeansDependOnFlyway() { @Test @WithResource(name = "db/changelog/db.changelog-master.yaml", content = "databaseChangeLog:") void jobRepositoryBeansDependOnLiquibase() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) - .withUserConfiguration(LiquibaseAutoConfiguration.class) + this.contextRunner + .withUserConfiguration(EmbeddedDataSourceConfiguration.class, LiquibaseAutoConfiguration.class) .withPropertyValues("spring.batch.initialize-schema=never") .run((context) -> { ConfigurableListableBeanFactory beanFactory = context.getBeanFactory(); @@ -431,8 +427,7 @@ void jobRepositoryBeansDependOnLiquibase() { @Test void whenTheUserDefinesTheirOwnBatchDatabaseInitializerThenTheAutoConfiguredInitializerBacksOff() { - this.contextRunner - .withUserConfiguration(TestConfiguration.class, CustomBatchDatabaseInitializerConfiguration.class) + this.contextRunner.withUserConfiguration(CustomBatchDatabaseInitializerConfiguration.class) .withConfiguration(AutoConfigurations.of(DataSourceAutoConfiguration.class, DataSourceTransactionManagerAutoConfiguration.class)) .run((context) -> assertThat(context).hasSingleBean(BatchDataSourceScriptDatabaseInitializer.class) @@ -442,7 +437,7 @@ void whenTheUserDefinesTheirOwnBatchDatabaseInitializerThenTheAutoConfiguredInit @Test void whenTheUserDefinesTheirOwnDatabaseInitializerThenTheAutoConfiguredBatchInitializerRemains() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, CustomDatabaseInitializerConfiguration.class) + this.contextRunner.withUserConfiguration(CustomDatabaseInitializerConfiguration.class) .withConfiguration(AutoConfigurations.of(DataSourceAutoConfiguration.class, DataSourceTransactionManagerAutoConfiguration.class)) .run((context) -> assertThat(context).hasSingleBean(BatchDataSourceScriptDatabaseInitializer.class) @@ -451,8 +446,9 @@ void whenTheUserDefinesTheirOwnDatabaseInitializerThenTheAutoConfiguredBatchInit @Test void conversionServiceCustomizersAreCalled() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) - .withUserConfiguration(ConversionServiceCustomizersConfiguration.class) + this.contextRunner + .withUserConfiguration(EmbeddedDataSourceConfiguration.class, + ConversionServiceCustomizersConfiguration.class) .run((context) -> { BatchConversionServiceCustomizer customizer = context.getBean("batchConversionServiceCustomizer", BatchConversionServiceCustomizer.class); @@ -487,7 +483,7 @@ void whenTheUserDefinesAJobNameThatDoesNotExistWithJobInstancesFailsFast() { JobLauncherApplicationRunner runner = createInstance(); runner.setJobs(Arrays.asList(mockJob("one"), mockJob("two"))); runner.setJobName("three"); - assertThatIllegalArgumentException().isThrownBy(runner::afterPropertiesSet) + assertThatIllegalStateException().isThrownBy(runner::afterPropertiesSet) .withMessage("No job found with name 'three'"); } @@ -495,14 +491,14 @@ void whenTheUserDefinesAJobNameThatDoesNotExistWithJobInstancesFailsFast() { void whenTheUserDefinesAJobNameThatDoesNotExistWithRegisteredJobFailsFast() { JobLauncherApplicationRunner runner = createInstance("one", "two"); runner.setJobName("three"); - assertThatIllegalArgumentException().isThrownBy(runner::afterPropertiesSet) + assertThatIllegalStateException().isThrownBy(runner::afterPropertiesSet) .withMessage("No job found with name 'three'"); } @Test void customExecutionContextSerializerIsUsed() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) - .withUserConfiguration(CustomExecutionContextConfiguration.class) + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class) + .withBean(ExecutionContextSerializer.class, Jackson2ExecutionContextStringSerializer::new) .run((context) -> { assertThat(context).hasSingleBean(Jackson2ExecutionContextStringSerializer.class); assertThat(context.getBean(SpringBootBatchConfiguration.class).getExecutionContextSerializer()) @@ -512,14 +508,47 @@ void customExecutionContextSerializerIsUsed() { @Test void defaultExecutionContextSerializerIsUsed() { - this.contextRunner.withUserConfiguration(TestConfiguration.class, EmbeddedDataSourceConfiguration.class) + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class).run((context) -> { + assertThat(context).doesNotHaveBean(ExecutionContextSerializer.class); + assertThat(context.getBean(SpringBootBatchConfiguration.class).getExecutionContextSerializer()) + .isInstanceOf(DefaultExecutionContextSerializer.class); + }); + } + + @Test + void customJdbcPropertiesIsUsed() { + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class) + .withPropertyValues("spring.batch.jdbc.validate-transaction-state:false", + "spring.batch.jdbc.isolation-level-for-create:READ_COMMITTED") .run((context) -> { - assertThat(context).doesNotHaveBean(ExecutionContextSerializer.class); - assertThat(context.getBean(SpringBootBatchConfiguration.class).getExecutionContextSerializer()) - .isInstanceOf(DefaultExecutionContextSerializer.class); + SpringBootBatchConfiguration configuration = context.getBean(SpringBootBatchConfiguration.class); + assertThat(configuration.getValidateTransactionState()).isEqualTo(false); + assertThat(configuration.getIsolationLevelForCreate()).isEqualTo(Isolation.READ_COMMITTED); + }); + + } + + @Test + void customJobParametersConverterIsUsed() { + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class) + .withBean(JobParametersConverter.class, JsonJobParametersConverter::new) + .withPropertyValues("spring.datasource.generate-unique-name=true") + .run((context) -> { + assertThat(context).hasSingleBean(JsonJobParametersConverter.class); + assertThat(context.getBean(SpringBootBatchConfiguration.class).getJobParametersConverter()) + .isInstanceOf(JsonJobParametersConverter.class); }); } + @Test + void defaultJobParametersConverterIsUsed() { + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class).run((context) -> { + assertThat(context).doesNotHaveBean(JobParametersConverter.class); + assertThat(context.getBean(SpringBootBatchConfiguration.class).getJobParametersConverter()) + .isInstanceOf(DefaultJobParametersConverter.class); + }); + } + private JobLauncherApplicationRunner createInstance(String... registeredJobNames) { JobLauncherApplicationRunner runner = new JobLauncherApplicationRunner(mock(JobLauncher.class), mock(JobExplorer.class), mock(JobRepository.class)); @@ -595,7 +624,7 @@ static class EmptyConfiguration { } @TestAutoConfigurationPackage(City.class) - static class TestConfiguration { + static class TestJpaConfiguration { } @@ -873,14 +902,4 @@ BatchConversionServiceCustomizer anotherBatchConversionServiceCustomizer() { } - @Configuration(proxyBeanMethods = false) - static class CustomExecutionContextConfiguration { - - @Bean - ExecutionContextSerializer executionContextSerializer() { - return new Jackson2ExecutionContextStringSerializer(); - } - - } - } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfigurationWithoutJpaTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfigurationWithoutJpaTests.java index 3377b5a449a8..b61b0fc00938 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfigurationWithoutJpaTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchAutoConfigurationWithoutJpaTests.java @@ -27,9 +27,9 @@ import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.TestAutoConfigurationPackage; import org.springframework.boot.autoconfigure.batch.BatchAutoConfiguration.SpringBootBatchConfiguration; +import org.springframework.boot.autoconfigure.batch.domain.City; import org.springframework.boot.autoconfigure.jdbc.DataSourceTransactionManagerAutoConfiguration; import org.springframework.boot.autoconfigure.jdbc.EmbeddedDataSourceConfiguration; -import org.springframework.boot.autoconfigure.orm.jpa.test.City; import org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration; import org.springframework.boot.sql.init.DatabaseInitializationMode; import org.springframework.boot.test.context.runner.ApplicationContextRunner; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchDataSourceScriptDatabaseInitializerTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchDataSourceScriptDatabaseInitializerTests.java index 724fda601a68..f22813c62341 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchDataSourceScriptDatabaseInitializerTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchDataSourceScriptDatabaseInitializerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -62,8 +62,8 @@ void getSettingsWithPlatformDoesNotTouchDataSource() { } @ParameterizedTest - @EnumSource(value = DatabaseDriver.class, mode = Mode.EXCLUDE, names = { "CLICKHOUSE", "FIREBIRD", "INFORMIX", - "JTDS", "PHOENIX", "REDSHIFT", "TERADATA", "TESTCONTAINERS", "UNKNOWN" }) + @EnumSource(value = DatabaseDriver.class, mode = Mode.EXCLUDE, names = { "AWS_WRAPPER", "CLICKHOUSE", "FIREBIRD", + "INFORMIX", "JTDS", "PHOENIX", "REDSHIFT", "TERADATA", "TESTCONTAINERS", "UNKNOWN" }) void batchSchemaCanBeLocated(DatabaseDriver driver) throws SQLException { DefaultResourceLoader resourceLoader = new DefaultResourceLoader(); BatchProperties properties = new BatchProperties(); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchPropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchPropertiesTests.java new file mode 100644 index 000000000000..a89fef9895a5 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/BatchPropertiesTests.java @@ -0,0 +1,47 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.batch; + +import org.junit.jupiter.api.Test; + +import org.springframework.batch.core.configuration.support.DefaultBatchConfiguration; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link BatchProperties}. + * + * @author Andy Wilkinson + */ +class BatchPropertiesTests { + + @Test + void validateTransactionStateDefaultMatchesSpringBatchDefault() { + assertThat(new BatchProperties().getJdbc().isValidateTransactionState()) + .isEqualTo(new TestBatchConfiguration().getValidateTransactionState()); + } + + static class TestBatchConfiguration extends DefaultBatchConfiguration { + + @Override + public boolean getValidateTransactionState() { + return super.getValidateTransactionState(); + } + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/domain/City.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/domain/City.java new file mode 100644 index 000000000000..8cb58ea87c11 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/batch/domain/City.java @@ -0,0 +1,78 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.batch.domain; + +import java.io.Serializable; + +import jakarta.persistence.Column; +import jakarta.persistence.Entity; +import jakarta.persistence.GeneratedValue; +import jakarta.persistence.Id; + +@Entity +public class City implements Serializable { + + private static final long serialVersionUID = 1L; + + @Id + @GeneratedValue + private Long id; + + @Column(nullable = false) + private String name; + + @Column(nullable = false) + private String state; + + @Column(nullable = false) + private String country; + + @Column(nullable = false) + private String map; + + protected City() { + } + + public City(String name, String state, String country, String map) { + this.name = name; + this.state = state; + this.country = country; + this.map = map; + } + + public String getName() { + return this.name; + } + + public String getState() { + return this.state; + } + + public String getCountry() { + return this.country; + } + + public String getMap() { + return this.map; + } + + @Override + public String toString() { + return getName() + "," + getState() + "," + getCountry(); + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/cache/CacheAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/cache/CacheAutoConfigurationTests.java index 5b576c3974ef..f0810d8db027 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/cache/CacheAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/cache/CacheAutoConfigurationTests.java @@ -51,7 +51,6 @@ import org.springframework.boot.autoconfigure.cache.support.MockCachingProvider.MockCacheManager; import org.springframework.boot.autoconfigure.hazelcast.HazelcastAutoConfiguration; import org.springframework.boot.test.context.assertj.AssertableApplicationContext; -import org.springframework.boot.testsupport.classpath.ClassPathExclusions; import org.springframework.boot.testsupport.classpath.resources.WithResource; import org.springframework.cache.Cache; import org.springframework.cache.Cache.ValueWrapper; @@ -97,7 +96,6 @@ * @author Mark Paluch * @author Ryon Day */ -@ClassPathExclusions("hazelcast-client-*.jar") class CacheAutoConfigurationTests extends AbstractCacheAutoConfigurationTests { @Test @@ -433,7 +431,7 @@ void jCacheCacheWithUnknownProvider() { @Test void jCacheCacheWithConfig() { String cachingProviderFqn = MockCachingProvider.class.getName(); - String configLocation = "org/springframework/boot/autoconfigure/hazelcast/hazelcast-specific.xml"; + String configLocation = "org/springframework/boot/autoconfigure/cache/hazelcast-specific.xml"; this.contextRunner.withUserConfiguration(JCacheCustomConfiguration.class) .withPropertyValues("spring.cache.type=jcache", "spring.cache.jcache.provider=" + cachingProviderFqn, "spring.cache.jcache.config=" + configLocation) @@ -453,7 +451,7 @@ void jCacheCacheWithWrongConfig() { "spring.cache.jcache.config=" + configLocation) .run((context) -> assertThat(context).getFailure() .isInstanceOf(BeanCreationException.class) - .hasMessageContaining("does not exist") + .hasMessageContaining("must exist") .hasMessageContaining(configLocation)); } @@ -521,7 +519,7 @@ void hazelcastCacheWithExistingHazelcastInstance() { @Test void hazelcastCacheWithHazelcastAutoConfiguration() { - String hazelcastConfig = "org/springframework/boot/autoconfigure/hazelcast/hazelcast-specific.xml"; + String hazelcastConfig = "org/springframework/boot/autoconfigure/cache/hazelcast-specific.xml"; this.contextRunner.withConfiguration(AutoConfigurations.of(HazelcastAutoConfiguration.class)) .withUserConfiguration(DefaultCacheConfiguration.class) .withPropertyValues("spring.cache.type=hazelcast", "spring.hazelcast.config=" + hazelcastConfig) diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBeanTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBeanTests.java index 01d852c0cdf0..53d9eba23925 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBeanTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBeanTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -52,6 +52,7 @@ * * @author Dave Syer * @author Stephane Nicoll + * @author Uladzislau Seuruk */ class ConditionalOnBeanTests { @@ -116,8 +117,8 @@ void testOnMissingBeanType() { @Test void withPropertyPlaceholderClassName() { this.contextRunner - .withUserConfiguration(PropertySourcesPlaceholderConfigurer.class, WithPropertyPlaceholderClassName.class, - OnBeanClassConfiguration.class) + .withUserConfiguration(PropertySourcesPlaceholderConfigurer.class, + WithPropertyPlaceholderClassNameConfiguration.class, OnBeanClassConfiguration.class) .withPropertyValues("mybeanclass=java.lang.String") .run((context) -> assertThat(context).hasNotFailed()); } @@ -165,7 +166,8 @@ void onBeanConditionOutputShouldNotContainConditionalOnMissingBeanClassInMessage @Test void conditionEvaluationConsidersChangeInTypeWhenBeanIsOverridden() { this.contextRunner.withAllowBeanDefinitionOverriding(true) - .withUserConfiguration(OriginalDefinition.class, OverridingDefinition.class, ConsumingConfiguration.class) + .withUserConfiguration(OriginalDefinitionConfiguration.class, OverridingDefinitionConfiguration.class, + ConsumingConfiguration.class) .run((context) -> { assertThat(context).hasBean("testBean"); assertThat(context).hasSingleBean(Integer.class); @@ -176,69 +178,73 @@ void conditionEvaluationConsidersChangeInTypeWhenBeanIsOverridden() { @Test void parameterizedContainerWhenValueIsOfMissingBeanDoesNotMatch() { this.contextRunner - .withUserConfiguration(ParameterizedWithoutCustomConfig.class, ParameterizedConditionWithValueConfig.class) - .run((context) -> assertThat(context).satisfies(exampleBeanRequirement("otherExampleBean"))); + .withUserConfiguration(ParameterizedWithoutCustomConfiguration.class, + ParameterizedConditionWithValueConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(ExampleBean.class, "otherExampleBean"))); } @Test void parameterizedContainerWhenValueIsOfExistingBeanMatches() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomConfig.class, ParameterizedConditionWithValueConfig.class) - .run((context) -> assertThat(context) - .satisfies(exampleBeanRequirement("customExampleBean", "conditionalCustomExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomConfiguration.class, + ParameterizedConditionWithValueConfiguration.class) + .run((context) -> assertThat(context).satisfies( + beansAndContainersNamed(ExampleBean.class, "customExampleBean", "conditionalCustomExampleBean"))); } @Test void parameterizedContainerWhenValueIsOfMissingBeanRegistrationDoesNotMatch() { this.contextRunner - .withUserConfiguration(ParameterizedWithoutCustomContainerConfig.class, - ParameterizedConditionWithValueConfig.class) - .run((context) -> assertThat(context).satisfies(exampleBeanRequirement("otherExampleBean"))); + .withUserConfiguration(ParameterizedWithoutCustomContainerConfiguration.class, + ParameterizedConditionWithValueConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(ExampleBean.class, "otherExampleBean"))); } @Test void parameterizedContainerWhenValueIsOfExistingBeanRegistrationMatches() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomContainerConfig.class, - ParameterizedConditionWithValueConfig.class) - .run((context) -> assertThat(context) - .satisfies(exampleBeanRequirement("customExampleBean", "conditionalCustomExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomContainerConfiguration.class, + ParameterizedConditionWithValueConfiguration.class) + .run((context) -> assertThat(context).satisfies( + beansAndContainersNamed(ExampleBean.class, "customExampleBean", "conditionalCustomExampleBean"))); } @Test void parameterizedContainerWhenReturnTypeIsOfExistingBeanMatches() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomConfig.class, - ParameterizedConditionWithReturnTypeConfig.class) - .run((context) -> assertThat(context) - .satisfies(exampleBeanRequirement("customExampleBean", "conditionalCustomExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomConfiguration.class, + ParameterizedConditionWithReturnTypeConfiguration.class) + .run((context) -> assertThat(context).satisfies( + beansAndContainersNamed(ExampleBean.class, "customExampleBean", "conditionalCustomExampleBean"))); } @Test void parameterizedContainerWhenReturnTypeIsOfExistingBeanRegistrationMatches() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomContainerConfig.class, - ParameterizedConditionWithReturnTypeConfig.class) - .run((context) -> assertThat(context) - .satisfies(exampleBeanRequirement("customExampleBean", "conditionalCustomExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomContainerConfiguration.class, + ParameterizedConditionWithReturnTypeConfiguration.class) + .run((context) -> assertThat(context).satisfies( + beansAndContainersNamed(ExampleBean.class, "customExampleBean", "conditionalCustomExampleBean"))); } @Test void parameterizedContainerWhenReturnRegistrationTypeIsOfExistingBeanMatches() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomConfig.class, - ParameterizedConditionWithReturnRegistrationTypeConfig.class) - .run((context) -> assertThat(context) - .satisfies(exampleBeanRequirement("customExampleBean", "conditionalCustomExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomConfiguration.class, + ParameterizedConditionWithReturnRegistrationTypeConfiguration.class) + .run((context) -> assertThat(context).satisfies( + beansAndContainersNamed(ExampleBean.class, "customExampleBean", "conditionalCustomExampleBean"))); } @Test void parameterizedContainerWhenReturnRegistrationTypeIsOfExistingBeanRegistrationMatches() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomContainerConfig.class, - ParameterizedConditionWithReturnRegistrationTypeConfig.class) - .run((context) -> assertThat(context) - .satisfies(exampleBeanRequirement("customExampleBean", "conditionalCustomExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomContainerConfiguration.class, + ParameterizedConditionWithReturnRegistrationTypeConfiguration.class) + .run((context) -> assertThat(context).satisfies( + beansAndContainersNamed(ExampleBean.class, "customExampleBean", "conditionalCustomExampleBean"))); } @Test @@ -257,7 +263,7 @@ void conditionalOnBeanNameMatchesNotAutowireCandidateBean() { @Test void conditionalOnAnnotatedBeanIgnoresNotAutowireCandidateBean() { this.contextRunner - .withUserConfiguration(AnnotatedNotAutowireCandidateConfig.class, OnAnnotationConfiguration.class) + .withUserConfiguration(AnnotatedNotAutowireCandidateConfiguration.class, OnAnnotationConfiguration.class) .run((context) -> assertThat(context).doesNotHaveBean("bar")); } @@ -284,13 +290,66 @@ void conditionalOnBeanNameMatchesNotDefaultCandidateBean() { @Test void conditionalOnAnnotatedBeanIgnoresNotDefaultCandidateBean() { this.contextRunner - .withUserConfiguration(AnnotatedNotDefaultCandidateConfig.class, OnAnnotationConfiguration.class) + .withUserConfiguration(AnnotatedNotDefaultCandidateConfiguration.class, OnAnnotationConfiguration.class) .run((context) -> assertThat(context).doesNotHaveBean("bar")); } - private Consumer exampleBeanRequirement(String... names) { + @Test + void genericWhenTypeArgumentMatches() { + this.contextRunner.withUserConfiguration(ParameterizedWithCustomGenericConfiguration.class, + GenericWithStringTypeArgumentsConfiguration.class, GenericWithIntegerTypeArgumentsConfiguration.class) + .run((context) -> assertThat(context).satisfies(beansAndContainersNamed(GenericExampleBean.class, + "customGenericExampleBean", "genericStringTypeArgumentsExampleBean"))); + } + + @Test + void genericWhenTypeArgumentWithValueMatches() { + this.contextRunner + .withUserConfiguration(GenericWithStringConfiguration.class, + TypeArgumentsConditionWithValueConfiguration.class) + .run((context) -> assertThat(context).satisfies(beansAndContainersNamed(GenericExampleBean.class, + "genericStringExampleBean", "genericStringWithValueExampleBean"))); + } + + @Test + void genericWithValueWhenSubclassTypeArgumentMatches() { + this.contextRunner + .withUserConfiguration(ParameterizedWithCustomGenericConfiguration.class, + TypeArgumentsConditionWithValueConfiguration.class) + .run((context) -> assertThat(context).satisfies(beansAndContainersNamed(GenericExampleBean.class, + "customGenericExampleBean", "genericStringWithValueExampleBean"))); + } + + @Test + void parameterizedContainerGenericWhenTypeArgumentNotMatches() { + this.contextRunner + .withUserConfiguration(GenericWithIntegerConfiguration.class, + TypeArgumentsConditionWithParameterizedContainerConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(GenericExampleBean.class, "genericIntegerExampleBean"))); + } + + @Test + void parameterizedContainerGenericWhenTypeArgumentMatches() { + this.contextRunner + .withUserConfiguration(GenericWithStringConfiguration.class, + TypeArgumentsConditionWithParameterizedContainerConfiguration.class) + .run((context) -> assertThat(context).satisfies(beansAndContainersNamed(GenericExampleBean.class, + "genericStringExampleBean", "parameterizedContainerGenericExampleBean"))); + } + + @Test + void parameterizedContainerGenericWhenSubclassTypeArgumentMatches() { + this.contextRunner + .withUserConfiguration(ParameterizedWithCustomGenericConfiguration.class, + TypeArgumentsConditionWithParameterizedContainerConfiguration.class) + .run((context) -> assertThat(context).satisfies(beansAndContainersNamed(GenericExampleBean.class, + "customGenericExampleBean", "parameterizedContainerGenericExampleBean"))); + } + + private Consumer beansAndContainersNamed(Class type, String... names) { return (context) -> { - String[] beans = context.getBeanNamesForType(ExampleBean.class); + String[] beans = context.getBeanNamesForType(type); String[] containers = context.getBeanNamesForType(TestParameterizedContainer.class); assertThat(StringUtils.concatenateStringArrays(beans, containers)).containsOnly(names); }; @@ -429,7 +488,7 @@ static class CombinedXmlConfiguration { @Configuration(proxyBeanMethods = false) @Import(WithPropertyPlaceholderClassNameRegistrar.class) - static class WithPropertyPlaceholderClassName { + static class WithPropertyPlaceholderClassNameConfiguration { } @@ -495,27 +554,8 @@ public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, } - static class ExampleFactoryBean implements FactoryBean { - - @Override - public ExampleBean getObject() { - return new ExampleBean("fromFactory"); - } - - @Override - public Class getObjectType() { - return ExampleBean.class; - } - - @Override - public boolean isSingleton() { - return false; - } - - } - @Configuration(proxyBeanMethods = false) - static class OriginalDefinition { + static class OriginalDefinitionConfiguration { @Bean String testBean() { @@ -526,7 +566,7 @@ String testBean() { @Configuration(proxyBeanMethods = false) @ConditionalOnBean(String.class) - static class OverridingDefinition { + static class OverridingDefinitionConfiguration { @Bean Integer testBean() { @@ -545,7 +585,7 @@ static class ConsumingConfiguration { } @Configuration(proxyBeanMethods = false) - static class ParameterizedWithCustomConfig { + static class ParameterizedWithCustomConfiguration { @Bean CustomExampleBean customExampleBean() { @@ -555,7 +595,7 @@ CustomExampleBean customExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedWithoutCustomConfig { + static class ParameterizedWithoutCustomConfiguration { @Bean OtherExampleBean otherExampleBean() { @@ -565,7 +605,7 @@ OtherExampleBean otherExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedWithoutCustomContainerConfig { + static class ParameterizedWithoutCustomContainerConfiguration { @Bean TestParameterizedContainer otherExampleBean() { @@ -575,7 +615,7 @@ TestParameterizedContainer otherExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedWithCustomContainerConfig { + static class ParameterizedWithCustomContainerConfiguration { @Bean TestParameterizedContainer customExampleBean() { @@ -585,7 +625,7 @@ TestParameterizedContainer customExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedConditionWithValueConfig { + static class ParameterizedConditionWithValueConfiguration { @Bean @ConditionalOnBean(value = CustomExampleBean.class, parameterizedContainer = TestParameterizedContainer.class) @@ -596,7 +636,7 @@ CustomExampleBean conditionalCustomExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedConditionWithReturnTypeConfig { + static class ParameterizedConditionWithReturnTypeConfiguration { @Bean @ConditionalOnBean(parameterizedContainer = TestParameterizedContainer.class) @@ -607,7 +647,7 @@ CustomExampleBean conditionalCustomExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedConditionWithReturnRegistrationTypeConfig { + static class ParameterizedConditionWithReturnRegistrationTypeConfiguration { @Bean @ConditionalOnBean(parameterizedContainer = TestParameterizedContainer.class) @@ -618,7 +658,7 @@ TestParameterizedContainer conditionalCustomExampleBean() { } @Configuration(proxyBeanMethods = false) - static class AnnotatedNotAutowireCandidateConfig { + static class AnnotatedNotAutowireCandidateConfiguration { @Bean(autowireCandidate = false) ExampleBean exampleBean() { @@ -628,7 +668,7 @@ ExampleBean exampleBean() { } @Configuration(proxyBeanMethods = false) - static class AnnotatedNotDefaultCandidateConfig { + static class AnnotatedNotDefaultCandidateConfiguration { @Bean(defaultCandidate = false) ExampleBean exampleBean() { @@ -637,6 +677,99 @@ ExampleBean exampleBean() { } + @Configuration(proxyBeanMethods = false) + static class ParameterizedWithCustomGenericConfiguration { + + @Bean + CustomGenericExampleBean customGenericExampleBean() { + return new CustomGenericExampleBean(); + } + + } + + @Configuration(proxyBeanMethods = false) + static class GenericWithStringConfiguration { + + @Bean + GenericExampleBean genericStringExampleBean() { + return new GenericExampleBean<>("genericStringExampleBean"); + } + + } + + @Configuration(proxyBeanMethods = false) + static class GenericWithStringTypeArgumentsConfiguration { + + @Bean + @ConditionalOnBean + GenericExampleBean genericStringTypeArgumentsExampleBean() { + return new GenericExampleBean<>("genericStringTypeArgumentsExampleBean"); + } + + } + + @Configuration(proxyBeanMethods = false) + static class GenericWithIntegerConfiguration { + + @Bean + GenericExampleBean genericIntegerExampleBean() { + return new GenericExampleBean<>(1_000); + } + + } + + @Configuration(proxyBeanMethods = false) + static class GenericWithIntegerTypeArgumentsConfiguration { + + @Bean + @ConditionalOnBean + GenericExampleBean genericIntegerTypeArgumentsExampleBean() { + return new GenericExampleBean<>(1_000); + } + + } + + @Configuration(proxyBeanMethods = false) + static class TypeArgumentsConditionWithValueConfiguration { + + @Bean + @ConditionalOnBean(GenericExampleBean.class) + GenericExampleBean genericStringWithValueExampleBean() { + return new GenericExampleBean<>("genericStringWithValueExampleBean"); + } + + } + + @Configuration(proxyBeanMethods = false) + static class TypeArgumentsConditionWithParameterizedContainerConfiguration { + + @Bean + @ConditionalOnBean(parameterizedContainer = TestParameterizedContainer.class) + TestParameterizedContainer> parameterizedContainerGenericExampleBean() { + return new TestParameterizedContainer<>(); + } + + } + + static class ExampleFactoryBean implements FactoryBean { + + @Override + public ExampleBean getObject() { + return new ExampleBean("fromFactory"); + } + + @Override + public Class getObjectType() { + return ExampleBean.class; + } + + @Override + public boolean isSingleton() { + return false; + } + + } + @TestAnnotation static class ExampleBean { @@ -669,6 +802,30 @@ static class OtherExampleBean extends ExampleBean { } + @TestAnnotation + static class GenericExampleBean { + + private final T value; + + GenericExampleBean(T value) { + this.value = value; + } + + @Override + public String toString() { + return String.valueOf(this.value); + } + + } + + static class CustomGenericExampleBean extends GenericExampleBean { + + CustomGenericExampleBean() { + super("custom subclass"); + } + + } + @Target({ ElementType.TYPE, ElementType.METHOD }) @Retention(RetentionPolicy.RUNTIME) @Documented diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBooleanPropertyTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBooleanPropertyTests.java new file mode 100644 index 000000000000..68b43ac685ed --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnBooleanPropertyTests.java @@ -0,0 +1,297 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.condition; + +import java.util.function.Consumer; +import java.util.stream.Collectors; + +import org.junit.jupiter.api.AfterEach; +import org.junit.jupiter.api.Test; + +import org.springframework.boot.WebApplicationType; +import org.springframework.boot.autoconfigure.condition.ConditionEvaluationReport.ConditionAndOutcomes; +import org.springframework.boot.builder.SpringApplicationBuilder; +import org.springframework.boot.test.util.TestPropertyValues; +import org.springframework.context.ConfigurableApplicationContext; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.core.env.ConfigurableEnvironment; +import org.springframework.core.env.StandardEnvironment; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; + +/** + * Tests for {@link ConditionalOnBooleanProperty @ConditionalOnBooleanProperty}. + * + * @author Phillip Webb + */ +class ConditionalOnBooleanPropertyTests { + + private ConfigurableApplicationContext context; + + private final ConfigurableEnvironment environment = new StandardEnvironment(); + + @AfterEach + void tearDown() { + if (this.context != null) { + this.context.close(); + } + } + + @Test + void defaultsWhenTrue() { + load(Defaults.class, "test=true"); + assertThat(this.context.containsBean("foo")).isTrue(); + } + + @Test + void defaultsWhenFalse() { + load(Defaults.class, "test=false"); + assertThat(this.context.containsBean("foo")).isFalse(); + } + + @Test + void defaultsWhenMissing() { + load(Defaults.class); + assertThat(this.context.containsBean("foo")).isFalse(); + } + + @Test + void havingValueTrueMatchIfMissingFalseWhenTrue() { + load(HavingValueTrueMatchIfMissingFalse.class, "test=true"); + assertThat(this.context.containsBean("foo")).isTrue(); + } + + @Test + void havingValueTrueMatchIfMissingFalseWhenFalse() { + load(HavingValueTrueMatchIfMissingFalse.class, "test=false"); + assertThat(this.context.containsBean("foo")).isFalse(); + } + + @Test + void havingValueTrueMatchIfMissingFalseWhenMissing() { + load(HavingValueTrueMatchIfMissingFalse.class); + assertThat(this.context.containsBean("foo")).isFalse(); + } + + @Test + void havingValueTrueMatchIfMissingTrueWhenTrue() { + load(HavingValueTrueMatchIfMissingTrue.class, "test=true"); + assertThat(this.context.containsBean("foo")).isTrue(); + } + + @Test + void havingValueTrueMatchIfMissingTrueWhenFalse() { + load(HavingValueTrueMatchIfMissingTrue.class, "test=false"); + assertThat(this.context.containsBean("foo")).isFalse(); + } + + @Test + void havingValueTrueMatchIfMissingTrueWhenMissing() { + load(HavingValueTrueMatchIfMissingTrue.class); + assertThat(this.context.containsBean("foo")).isTrue(); + } + + @Test + void havingValueFalseMatchIfMissingFalseWhenTrue() { + load(HavingValueFalseMatchIfMissingFalse.class, "test=true"); + assertThat(this.context.containsBean("foo")).isFalse(); + } + + @Test + void havingValueFalseMatchIfMissingFalseWhenFalse() { + load(HavingValueFalseMatchIfMissingFalse.class, "test=false"); + assertThat(this.context.containsBean("foo")).isTrue(); + } + + @Test + void havingValueFalseMatchIfMissingFalseWhenMissing() { + load(HavingValueFalseMatchIfMissingFalse.class); + assertThat(this.context.containsBean("foo")).isFalse(); + } + + @Test + void havingValueFalseMatchIfMissingTrueWhenTrue() { + load(HavingValueFalseMatchIfMissingTrue.class, "test=true"); + assertThat(this.context.containsBean("foo")).isFalse(); + } + + @Test + void havingValueFalseMatchIfMissingTrueWhenFalse() { + load(HavingValueFalseMatchIfMissingTrue.class, "test=false"); + assertThat(this.context.containsBean("foo")).isTrue(); + } + + @Test + void havingValueFalseMatchIfMissingTrueWhenMissing() { + load(HavingValueFalseMatchIfMissingTrue.class); + assertThat(this.context.containsBean("foo")).isTrue(); + } + + @Test + void withPrefix() { + load(HavingValueFalseMatchIfMissingTrue.class, "foo.test=true"); + assertThat(this.context.containsBean("foo")).isTrue(); + } + + @Test + void nameOrValueMustBeSpecified() { + assertThatIllegalStateException().isThrownBy(() -> load(NoNameOrValueAttribute.class, "some.property")) + .satisfies(causeMessageContaining( + "The name or value attribute of @ConditionalOnBooleanProperty must be specified")); + } + + @Test + void nameAndValueMustNotBeSpecified() { + assertThatIllegalStateException().isThrownBy(() -> load(NameAndValueAttribute.class, "some.property")) + .satisfies(causeMessageContaining( + "The name and value attributes of @ConditionalOnBooleanProperty are exclusive")); + } + + @Test + void conditionReportWhenMatched() { + load(Defaults.class, "test=true"); + assertThat(this.context.containsBean("foo")).isTrue(); + assertThat(getConditionEvaluationReport()).contains("@ConditionalOnBooleanProperty (test=true) matched"); + } + + @Test + void conditionReportWhenDoesNotMatch() { + load(Defaults.class, "test=false"); + assertThat(this.context.containsBean("foo")).isFalse(); + assertThat(getConditionEvaluationReport()) + .contains("@ConditionalOnBooleanProperty (test=true) found different value in property 'test'"); + } + + @Test + void repeatablePropertiesConditionReportWhenMatched() { + load(RepeatablePropertiesRequiredConfiguration.class, "property1=true", "property2=true"); + assertThat(this.context.containsBean("foo")).isTrue(); + String report = getConditionEvaluationReport(); + assertThat(report).contains("@ConditionalOnBooleanProperty (property1=true) matched"); + assertThat(report).contains("@ConditionalOnBooleanProperty (property2=true) matched"); + } + + @Test + void repeatablePropertiesConditionReportWhenDoesNotMatch() { + load(RepeatablePropertiesRequiredConfiguration.class, "property1=true"); + assertThat(getConditionEvaluationReport()) + .contains("@ConditionalOnBooleanProperty (property2=true) did not find property 'property2'"); + } + + private Consumer causeMessageContaining(String message) { + return (ex) -> assertThat(ex.getCause()).hasMessageContaining(message); + } + + private String getConditionEvaluationReport() { + return ConditionEvaluationReport.get(this.context.getBeanFactory()) + .getConditionAndOutcomesBySource() + .values() + .stream() + .flatMap(ConditionAndOutcomes::stream) + .map(Object::toString) + .collect(Collectors.joining("\n")); + } + + private void load(Class config, String... environment) { + TestPropertyValues.of(environment).applyTo(this.environment); + this.context = new SpringApplicationBuilder(config).environment(this.environment) + .web(WebApplicationType.NONE) + .run(); + } + + abstract static class BeanConfiguration { + + @Bean + String foo() { + return "foo"; + } + + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnBooleanProperty("test") + static class Defaults extends BeanConfiguration { + + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnBooleanProperty(name = "test", havingValue = true, matchIfMissing = false) + static class HavingValueTrueMatchIfMissingFalse extends BeanConfiguration { + + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnBooleanProperty(name = "test", havingValue = true, matchIfMissing = true) + static class HavingValueTrueMatchIfMissingTrue extends BeanConfiguration { + + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnBooleanProperty(name = "test", havingValue = false, matchIfMissing = false) + static class HavingValueFalseMatchIfMissingFalse extends BeanConfiguration { + + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnBooleanProperty(name = "test", havingValue = false, matchIfMissing = true) + static class HavingValueFalseMatchIfMissingTrue extends BeanConfiguration { + + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnBooleanProperty(prefix = "foo", name = "test") + static class WithPrefix extends BeanConfiguration { + + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnBooleanProperty + static class NoNameOrValueAttribute { + + @Bean + String foo() { + return "foo"; + } + + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnBooleanProperty(value = "x", name = "y") + static class NameAndValueAttribute { + + @Bean + String foo() { + return "foo"; + } + + } + + @Configuration(proxyBeanMethods = false) + @ConditionalOnBooleanProperty("property1") + @ConditionalOnBooleanProperty("property2") + static class RepeatablePropertiesRequiredConfiguration { + + @Bean + String foo() { + return "foo"; + } + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnMissingBeanTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnMissingBeanTests.java index b908dc67a00e..7b5e44da3e65 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnMissingBeanTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnMissingBeanTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -61,6 +61,7 @@ * @author Phillip Webb * @author Jakub Kubrynski * @author Andy Wilkinson + * @author Uladzislau Seuruk */ @SuppressWarnings("resource") class ConditionalOnMissingBeanTests { @@ -100,7 +101,7 @@ void testNameAndTypeOnMissingBeanCondition() { void hierarchyConsidered() { this.contextRunner.withUserConfiguration(FooConfiguration.class) .run((parent) -> new ApplicationContextRunner().withParent(parent) - .withUserConfiguration(HierarchyConsidered.class) + .withUserConfiguration(HierarchyConsideredConfiguration.class) .run((context) -> assertThat(context.containsLocalBean("bar")).isFalse())); } @@ -108,13 +109,13 @@ void hierarchyConsidered() { void hierarchyNotConsidered() { this.contextRunner.withUserConfiguration(FooConfiguration.class) .run((parent) -> new ApplicationContextRunner().withParent(parent) - .withUserConfiguration(HierarchyNotConsidered.class) + .withUserConfiguration(HierarchyNotConsideredConfiguration.class) .run((context) -> assertThat(context.containsLocalBean("bar")).isTrue())); } @Test void impliedOnBeanMethod() { - this.contextRunner.withUserConfiguration(ExampleBeanConfiguration.class, ImpliedOnBeanMethod.class) + this.contextRunner.withUserConfiguration(ExampleBeanConfiguration.class, ImpliedOnBeanMethodConfiguration.class) .run((context) -> assertThat(context).hasSingleBean(ExampleBean.class)); } @@ -173,7 +174,8 @@ void testOnMissingBeanConditionOutputShouldNotContainConditionalOnBeanClassInMes @Test void testOnMissingBeanConditionWithFactoryBean() { this.contextRunner - .withUserConfiguration(FactoryBeanConfiguration.class, ConditionalOnMissingBeanProducedByFactoryBean.class, + .withUserConfiguration(FactoryBeanConfiguration.class, + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, PropertyPlaceholderAutoConfiguration.class) .run((context) -> assertThat(context.getBean(ExampleBean.class)).hasToString("fromFactory")); } @@ -182,7 +184,8 @@ void testOnMissingBeanConditionWithFactoryBean() { void testOnMissingBeanConditionWithComponentScannedFactoryBean() { this.contextRunner .withUserConfiguration(ComponentScannedFactoryBeanBeanMethodConfiguration.class, - ConditionalOnMissingBeanProducedByFactoryBean.class, PropertyPlaceholderAutoConfiguration.class) + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, + PropertyPlaceholderAutoConfiguration.class) .run((context) -> assertThat(context.getBean(ScanBean.class)).hasToString("fromFactory")); } @@ -190,7 +193,8 @@ void testOnMissingBeanConditionWithComponentScannedFactoryBean() { void testOnMissingBeanConditionWithComponentScannedFactoryBeanWithBeanMethodArguments() { this.contextRunner .withUserConfiguration(ComponentScannedFactoryBeanBeanMethodWithArgumentsConfiguration.class, - ConditionalOnMissingBeanProducedByFactoryBean.class, PropertyPlaceholderAutoConfiguration.class) + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, + PropertyPlaceholderAutoConfiguration.class) .run((context) -> assertThat(context.getBean(ScanBean.class)).hasToString("fromFactory")); } @@ -198,7 +202,8 @@ void testOnMissingBeanConditionWithComponentScannedFactoryBeanWithBeanMethodArgu void testOnMissingBeanConditionWithFactoryBeanWithBeanMethodArguments() { this.contextRunner .withUserConfiguration(FactoryBeanWithBeanMethodArgumentsConfiguration.class, - ConditionalOnMissingBeanProducedByFactoryBean.class, PropertyPlaceholderAutoConfiguration.class) + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, + PropertyPlaceholderAutoConfiguration.class) .withPropertyValues("theValue=foo") .run((context) -> assertThat(context.getBean(ExampleBean.class)).hasToString("fromFactory")); } @@ -207,7 +212,8 @@ void testOnMissingBeanConditionWithFactoryBeanWithBeanMethodArguments() { void testOnMissingBeanConditionWithConcreteFactoryBean() { this.contextRunner .withUserConfiguration(ConcreteFactoryBeanConfiguration.class, - ConditionalOnMissingBeanProducedByFactoryBean.class, PropertyPlaceholderAutoConfiguration.class) + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, + PropertyPlaceholderAutoConfiguration.class) .run((context) -> assertThat(context.getBean(ExampleBean.class)).hasToString("fromFactory")); } @@ -216,7 +222,8 @@ void testOnMissingBeanConditionWithUnhelpfulFactoryBean() { // We could not tell that the FactoryBean would ultimately create an ExampleBean this.contextRunner .withUserConfiguration(UnhelpfulFactoryBeanConfiguration.class, - ConditionalOnMissingBeanProducedByFactoryBean.class, PropertyPlaceholderAutoConfiguration.class) + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, + PropertyPlaceholderAutoConfiguration.class) .run((context) -> assertThat(context).getBeans(ExampleBean.class).hasSize(2)); } @@ -224,7 +231,8 @@ void testOnMissingBeanConditionWithUnhelpfulFactoryBean() { void testOnMissingBeanConditionWithRegisteredFactoryBean() { this.contextRunner .withUserConfiguration(RegisteredFactoryBeanConfiguration.class, - ConditionalOnMissingBeanProducedByFactoryBean.class, PropertyPlaceholderAutoConfiguration.class) + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, + PropertyPlaceholderAutoConfiguration.class) .run((context) -> assertThat(context.getBean(ExampleBean.class)).hasToString("fromFactory")); } @@ -232,7 +240,8 @@ void testOnMissingBeanConditionWithRegisteredFactoryBean() { void testOnMissingBeanConditionWithNonspecificFactoryBeanWithClassAttribute() { this.contextRunner .withUserConfiguration(NonspecificFactoryBeanClassAttributeConfiguration.class, - ConditionalOnMissingBeanProducedByFactoryBean.class, PropertyPlaceholderAutoConfiguration.class) + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, + PropertyPlaceholderAutoConfiguration.class) .run((context) -> assertThat(context.getBean(ExampleBean.class)).hasToString("fromFactory")); } @@ -240,7 +249,8 @@ void testOnMissingBeanConditionWithNonspecificFactoryBeanWithClassAttribute() { void testOnMissingBeanConditionWithNonspecificFactoryBeanWithStringAttribute() { this.contextRunner .withUserConfiguration(NonspecificFactoryBeanStringAttributeConfiguration.class, - ConditionalOnMissingBeanProducedByFactoryBean.class, PropertyPlaceholderAutoConfiguration.class) + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, + PropertyPlaceholderAutoConfiguration.class) .run((context) -> assertThat(context.getBean(ExampleBean.class)).hasToString("fromFactory")); } @@ -248,15 +258,16 @@ void testOnMissingBeanConditionWithNonspecificFactoryBeanWithStringAttribute() { void testOnMissingBeanConditionWithFactoryBeanInXml() { this.contextRunner .withUserConfiguration(FactoryBeanXmlConfiguration.class, - ConditionalOnMissingBeanProducedByFactoryBean.class, PropertyPlaceholderAutoConfiguration.class) + ConditionalOnMissingBeanProducedByFactoryBeanConfiguration.class, + PropertyPlaceholderAutoConfiguration.class) .run((context) -> assertThat(context.getBean(ExampleBean.class)).hasToString("fromFactory")); } @Test void testOnMissingBeanConditionWithIgnoredSubclass() { this.contextRunner - .withUserConfiguration(CustomExampleBeanConfiguration.class, ConditionalOnIgnoredSubclass.class, - PropertyPlaceholderAutoConfiguration.class) + .withUserConfiguration(CustomExampleBeanConfiguration.class, + ConditionalOnIgnoredSubclassConfiguration.class, PropertyPlaceholderAutoConfiguration.class) .run((context) -> { assertThat(context).getBeans(ExampleBean.class).hasSize(2); assertThat(context).getBeans(CustomExampleBean.class).hasSize(1); @@ -266,8 +277,8 @@ void testOnMissingBeanConditionWithIgnoredSubclass() { @Test void testOnMissingBeanConditionWithIgnoredSubclassByName() { this.contextRunner - .withUserConfiguration(CustomExampleBeanConfiguration.class, ConditionalOnIgnoredSubclassByName.class, - PropertyPlaceholderAutoConfiguration.class) + .withUserConfiguration(CustomExampleBeanConfiguration.class, + ConditionalOnIgnoredSubclassByNameConfiguration.class, PropertyPlaceholderAutoConfiguration.class) .run((context) -> { assertThat(context).getBeans(ExampleBean.class).hasSize(2); assertThat(context).getBeans(CustomExampleBean.class).hasSize(1); @@ -304,89 +315,97 @@ void beanProducedByFactoryBeanIsConsideredWhenMatchingOnAnnotation() { @Test void parameterizedContainerWhenValueIsOfMissingBeanMatches() { this.contextRunner - .withUserConfiguration(ParameterizedWithoutCustomConfig.class, ParameterizedConditionWithValueConfig.class) - .run((context) -> assertThat(context) - .satisfies(exampleBeanRequirement("otherExampleBean", "conditionalCustomExampleBean"))); + .withUserConfiguration(ParameterizedWithoutCustomConfiguration.class, + ParameterizedConditionWithValueConfiguration.class) + .run((context) -> assertThat(context).satisfies( + beansAndContainersNamed(ExampleBean.class, "otherExampleBean", "conditionalCustomExampleBean"))); } @Test void parameterizedContainerWhenValueIsOfExistingBeanDoesNotMatch() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomConfig.class, ParameterizedConditionWithValueConfig.class) - .run((context) -> assertThat(context).satisfies(exampleBeanRequirement("customExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomConfiguration.class, + ParameterizedConditionWithValueConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(ExampleBean.class, "customExampleBean"))); } @Test void parameterizedContainerWhenValueIsOfMissingBeanRegistrationMatches() { this.contextRunner - .withUserConfiguration(ParameterizedWithoutCustomContainerConfig.class, - ParameterizedConditionWithValueConfig.class) - .run((context) -> assertThat(context) - .satisfies(exampleBeanRequirement("otherExampleBean", "conditionalCustomExampleBean"))); + .withUserConfiguration(ParameterizedWithoutCustomContainerConfiguration.class, + ParameterizedConditionWithValueConfiguration.class) + .run((context) -> assertThat(context).satisfies( + beansAndContainersNamed(ExampleBean.class, "otherExampleBean", "conditionalCustomExampleBean"))); } @Test void parameterizedContainerWhenValueIsOfExistingBeanRegistrationDoesNotMatch() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomContainerConfig.class, - ParameterizedConditionWithValueConfig.class) - .run((context) -> assertThat(context).satisfies(exampleBeanRequirement("customExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomContainerConfiguration.class, + ParameterizedConditionWithValueConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(ExampleBean.class, "customExampleBean"))); } @Test void parameterizedContainerWhenReturnTypeIsOfExistingBeanDoesNotMatch() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomConfig.class, - ParameterizedConditionWithReturnTypeConfig.class) - .run((context) -> assertThat(context).satisfies(exampleBeanRequirement("customExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomConfiguration.class, + ParameterizedConditionWithReturnTypeConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(ExampleBean.class, "customExampleBean"))); } @Test void parameterizedContainerWhenReturnTypeIsOfExistingBeanRegistrationDoesNotMatch() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomContainerConfig.class, - ParameterizedConditionWithReturnTypeConfig.class) - .run((context) -> assertThat(context).satisfies(exampleBeanRequirement("customExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomContainerConfiguration.class, + ParameterizedConditionWithReturnTypeConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(ExampleBean.class, "customExampleBean"))); } @Test void parameterizedContainerWhenReturnRegistrationTypeIsOfExistingBeanDoesNotMatch() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomConfig.class, - ParameterizedConditionWithReturnRegistrationTypeConfig.class) - .run((context) -> assertThat(context).satisfies(exampleBeanRequirement("customExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomConfiguration.class, + ParameterizedConditionWithReturnRegistrationTypeConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(ExampleBean.class, "customExampleBean"))); } @Test void parameterizedContainerWhenReturnRegistrationTypeIsOfExistingBeanRegistrationDoesNotMatch() { this.contextRunner - .withUserConfiguration(ParameterizedWithCustomContainerConfig.class, - ParameterizedConditionWithReturnRegistrationTypeConfig.class) - .run((context) -> assertThat(context).satisfies(exampleBeanRequirement("customExampleBean"))); + .withUserConfiguration(ParameterizedWithCustomContainerConfiguration.class, + ParameterizedConditionWithReturnRegistrationTypeConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(ExampleBean.class, "customExampleBean"))); } @Test void typeBasedMatchingIgnoresBeanThatIsNotAutowireCandidate() { - this.contextRunner.withUserConfiguration(NotAutowireCandidateConfig.class, OnBeanTypeConfiguration.class) + this.contextRunner.withUserConfiguration(NotAutowireCandidateConfiguration.class, OnBeanTypeConfiguration.class) .run((context) -> assertThat(context).hasBean("bar")); } @Test void nameBasedMatchingConsidersBeanThatIsNotAutowireCandidate() { - this.contextRunner.withUserConfiguration(NotAutowireCandidateConfig.class, OnBeanNameConfiguration.class) + this.contextRunner.withUserConfiguration(NotAutowireCandidateConfiguration.class, OnBeanNameConfiguration.class) .run((context) -> assertThat(context).doesNotHaveBean("bar")); } @Test void annotationBasedMatchingIgnoresBeanThatIsNotAutowireCandidateBean() { this.contextRunner - .withUserConfiguration(AnnotatedNotAutowireCandidateConfig.class, OnAnnotationConfiguration.class) + .withUserConfiguration(AnnotatedNotAutowireCandidateConfiguration.class, OnAnnotationConfiguration.class) .run((context) -> assertThat(context).hasBean("bar")); } @Test void typeBasedMatchingIgnoresBeanThatIsNotDefaultCandidate() { - this.contextRunner.withUserConfiguration(NotDefaultCandidateConfig.class, OnBeanTypeConfiguration.class) + this.contextRunner.withUserConfiguration(NotDefaultCandidateConfiguration.class, OnBeanTypeConfiguration.class) .run((context) -> assertThat(context).hasBean("bar")); } @@ -394,27 +413,99 @@ void typeBasedMatchingIgnoresBeanThatIsNotDefaultCandidate() { void typeBasedMatchingIgnoresFactoryBeanThatIsNotDefaultCandidate() { this.contextRunner .withUserConfiguration(NotDefaultCandidateFactoryBeanConfiguration.class, - ConditionalOnMissingFactoryBean.class) + ConditionalOnMissingFactoryBeanConfiguration.class) .run((context) -> assertThat(context).hasBean("&exampleFactoryBean") .hasBean("&additionalExampleFactoryBean")); } @Test void nameBasedMatchingConsidersBeanThatIsNotDefaultCandidate() { - this.contextRunner.withUserConfiguration(NotDefaultCandidateConfig.class, OnBeanNameConfiguration.class) + this.contextRunner.withUserConfiguration(NotDefaultCandidateConfiguration.class, OnBeanNameConfiguration.class) .run((context) -> assertThat(context).doesNotHaveBean("bar")); } @Test void annotationBasedMatchingIgnoresBeanThatIsNotDefaultCandidateBean() { this.contextRunner - .withUserConfiguration(AnnotatedNotDefaultCandidateConfig.class, OnAnnotationConfiguration.class) + .withUserConfiguration(AnnotatedNotDefaultCandidateConfiguration.class, OnAnnotationConfiguration.class) .run((context) -> assertThat(context).hasBean("bar")); } - private Consumer exampleBeanRequirement(String... names) { + @Test + void genericWhenTypeArgumentNotMatches() { + this.contextRunner + .withUserConfiguration(GenericWithStringTypeArgumentsConfiguration.class, + GenericWithIntegerTypeArgumentsConfiguration.class) + .run((context) -> assertThat(context).satisfies(beansAndContainersNamed(GenericExampleBean.class, + "genericStringExampleBean", "genericIntegerExampleBean"))); + } + + @Test + void genericWhenSubclassTypeArgumentMatches() { + this.contextRunner + .withUserConfiguration(ParameterizedWithCustomGenericConfiguration.class, + GenericWithStringTypeArgumentsConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(GenericExampleBean.class, "customGenericExampleBean"))); + } + + @Test + void genericWhenSubclassTypeArgumentNotMatches() { + this.contextRunner + .withUserConfiguration(ParameterizedWithCustomGenericConfiguration.class, + GenericWithIntegerTypeArgumentsConfiguration.class) + .run((context) -> assertThat(context).satisfies(beansAndContainersNamed(GenericExampleBean.class, + "customGenericExampleBean", "genericIntegerExampleBean"))); + } + + @Test + void genericWhenTypeArgumentWithValueMatches() { + this.contextRunner + .withUserConfiguration(GenericWithStringTypeArgumentsConfiguration.class, + TypeArgumentsConditionWithValueConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(GenericExampleBean.class, "genericStringExampleBean"))); + } + + @Test + void genericWithValueWhenSubclassTypeArgumentMatches() { + this.contextRunner + .withUserConfiguration(ParameterizedWithCustomGenericConfiguration.class, + TypeArgumentsConditionWithValueConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(GenericExampleBean.class, "customGenericExampleBean"))); + } + + @Test + void parameterizedContainerGenericWhenTypeArgumentNotMatches() { + this.contextRunner + .withUserConfiguration(GenericWithIntegerTypeArgumentsConfiguration.class, + TypeArgumentsConditionWithParameterizedContainerConfiguration.class) + .run((context) -> assertThat(context).satisfies(beansAndContainersNamed(GenericExampleBean.class, + "genericIntegerExampleBean", "parameterizedContainerGenericExampleBean"))); + } + + @Test + void parameterizedContainerGenericWhenTypeArgumentMatches() { + this.contextRunner + .withUserConfiguration(GenericWithStringTypeArgumentsConfiguration.class, + TypeArgumentsConditionWithParameterizedContainerConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(GenericExampleBean.class, "genericStringExampleBean"))); + } + + @Test + void parameterizedContainerGenericWhenSubclassTypeArgumentMatches() { + this.contextRunner + .withUserConfiguration(ParameterizedWithCustomGenericConfiguration.class, + TypeArgumentsConditionWithParameterizedContainerConfiguration.class) + .run((context) -> assertThat(context) + .satisfies(beansAndContainersNamed(GenericExampleBean.class, "customGenericExampleBean"))); + } + + private Consumer beansAndContainersNamed(Class type, String... names) { return (context) -> { - String[] beans = context.getBeanNamesForType(ExampleBean.class); + String[] beans = context.getBeanNamesForType(type); String[] containers = context.getBeanNamesForType(TestParameterizedContainer.class); assertThat(StringUtils.concatenateStringArrays(beans, containers)).containsOnly(names); }; @@ -584,7 +675,7 @@ static class FactoryBeanXmlConfiguration { } @Configuration(proxyBeanMethods = false) - static class ConditionalOnMissingBeanProducedByFactoryBean { + static class ConditionalOnMissingBeanProducedByFactoryBeanConfiguration { @Bean @ConditionalOnMissingBean @@ -595,7 +686,7 @@ ExampleBean createExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ConditionalOnMissingFactoryBean { + static class ConditionalOnMissingFactoryBeanConfiguration { @Bean @ConditionalOnMissingBean @@ -606,7 +697,7 @@ ExampleFactoryBean additionalExampleFactoryBean() { } @Configuration(proxyBeanMethods = false) - static class ConditionalOnIgnoredSubclass { + static class ConditionalOnIgnoredSubclassConfiguration { @Bean @ConditionalOnMissingBean(ignored = CustomExampleBean.class) @@ -617,7 +708,7 @@ ExampleBean exampleBean() { } @Configuration(proxyBeanMethods = false) - static class ConditionalOnIgnoredSubclassByName { + static class ConditionalOnIgnoredSubclassByNameConfiguration { @Bean @ConditionalOnMissingBean( @@ -683,7 +774,7 @@ String foo() { } @Configuration(proxyBeanMethods = false) - static class NotAutowireCandidateConfig { + static class NotAutowireCandidateConfiguration { @Bean(autowireCandidate = false) String foo() { @@ -693,7 +784,7 @@ String foo() { } @Configuration(proxyBeanMethods = false) - static class NotDefaultCandidateConfig { + static class NotDefaultCandidateConfiguration { @Bean(defaultCandidate = false) String foo() { @@ -704,7 +795,7 @@ String foo() { @Configuration(proxyBeanMethods = false) @ConditionalOnMissingBean(name = "foo") - static class HierarchyConsidered { + static class HierarchyConsideredConfiguration { @Bean String bar() { @@ -715,7 +806,7 @@ String bar() { @Configuration(proxyBeanMethods = false) @ConditionalOnMissingBean(name = "foo", search = SearchStrategy.CURRENT) - static class HierarchyNotConsidered { + static class HierarchyNotConsideredConfiguration { @Bean String bar() { @@ -751,7 +842,7 @@ UnrelatedExampleBean unrelatedExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ImpliedOnBeanMethod { + static class ImpliedOnBeanMethodConfiguration { @Bean @ConditionalOnMissingBean @@ -761,54 +852,8 @@ ExampleBean exampleBean2() { } - static class ExampleFactoryBean implements FactoryBean { - - ExampleFactoryBean(String value) { - Assert.state(!value.contains("$"), "value should not contain '$'"); - } - - @Override - public ExampleBean getObject() { - return new ExampleBean("fromFactory"); - } - - @Override - public Class getObjectType() { - return ExampleBean.class; - } - - @Override - public boolean isSingleton() { - return false; - } - - } - - static class NonspecificFactoryBean implements FactoryBean { - - NonspecificFactoryBean(String value) { - Assert.state(!value.contains("$"), "value should not contain '$'"); - } - - @Override - public ExampleBean getObject() { - return new ExampleBean("fromFactory"); - } - - @Override - public Class getObjectType() { - return ExampleBean.class; - } - - @Override - public boolean isSingleton() { - return false; - } - - } - @Configuration(proxyBeanMethods = false) - static class ParameterizedWithCustomConfig { + static class ParameterizedWithCustomConfiguration { @Bean CustomExampleBean customExampleBean() { @@ -818,7 +863,7 @@ CustomExampleBean customExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedWithoutCustomConfig { + static class ParameterizedWithoutCustomConfiguration { @Bean OtherExampleBean otherExampleBean() { @@ -828,7 +873,7 @@ OtherExampleBean otherExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedWithoutCustomContainerConfig { + static class ParameterizedWithoutCustomContainerConfiguration { @Bean TestParameterizedContainer otherExampleBean() { @@ -838,7 +883,7 @@ TestParameterizedContainer otherExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedWithCustomContainerConfig { + static class ParameterizedWithCustomContainerConfiguration { @Bean TestParameterizedContainer customExampleBean() { @@ -848,7 +893,7 @@ TestParameterizedContainer customExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedConditionWithValueConfig { + static class ParameterizedConditionWithValueConfiguration { @Bean @ConditionalOnMissingBean(parameterizedContainer = TestParameterizedContainer.class) @@ -859,7 +904,7 @@ CustomExampleBean conditionalCustomExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedConditionWithReturnTypeConfig { + static class ParameterizedConditionWithReturnTypeConfiguration { @Bean @ConditionalOnMissingBean(parameterizedContainer = TestParameterizedContainer.class) @@ -870,7 +915,7 @@ CustomExampleBean conditionalCustomExampleBean() { } @Configuration(proxyBeanMethods = false) - static class ParameterizedConditionWithReturnRegistrationTypeConfig { + static class ParameterizedConditionWithReturnRegistrationTypeConfiguration { @Bean @ConditionalOnMissingBean(parameterizedContainer = TestParameterizedContainer.class) @@ -881,7 +926,7 @@ TestParameterizedContainer conditionalCustomExampleBean() { } @Configuration(proxyBeanMethods = false) - static class AnnotatedNotAutowireCandidateConfig { + static class AnnotatedNotAutowireCandidateConfiguration { @Bean(autowireCandidate = false) ExampleBean exampleBean() { @@ -891,7 +936,7 @@ ExampleBean exampleBean() { } @Configuration(proxyBeanMethods = false) - static class AnnotatedNotDefaultCandidateConfig { + static class AnnotatedNotDefaultCandidateConfiguration { @Bean(autowireCandidate = false) ExampleBean exampleBean() { @@ -900,6 +945,106 @@ ExampleBean exampleBean() { } + @Configuration(proxyBeanMethods = false) + static class ParameterizedWithCustomGenericConfiguration { + + @Bean + CustomGenericExampleBean customGenericExampleBean() { + return new CustomGenericExampleBean(); + } + + } + + @Configuration(proxyBeanMethods = false) + static class GenericWithStringTypeArgumentsConfiguration { + + @Bean + @ConditionalOnMissingBean + GenericExampleBean genericStringExampleBean() { + return new GenericExampleBean<>("genericStringExampleBean"); + } + + } + + @Configuration(proxyBeanMethods = false) + static class GenericWithIntegerTypeArgumentsConfiguration { + + @Bean + @ConditionalOnMissingBean + GenericExampleBean genericIntegerExampleBean() { + return new GenericExampleBean<>(1_000); + } + + } + + @Configuration(proxyBeanMethods = false) + static class TypeArgumentsConditionWithValueConfiguration { + + @Bean + @ConditionalOnMissingBean(GenericExampleBean.class) + GenericExampleBean genericStringWithValueExampleBean() { + return new GenericExampleBean<>("genericStringWithValueExampleBean"); + } + + } + + @Configuration(proxyBeanMethods = false) + static class TypeArgumentsConditionWithParameterizedContainerConfiguration { + + @Bean + @ConditionalOnMissingBean(parameterizedContainer = TestParameterizedContainer.class) + TestParameterizedContainer> parameterizedContainerGenericExampleBean() { + return new TestParameterizedContainer<>(); + } + + } + + static class ExampleFactoryBean implements FactoryBean { + + ExampleFactoryBean(String value) { + Assert.state(!value.contains("$"), "value should not contain '$'"); + } + + @Override + public ExampleBean getObject() { + return new ExampleBean("fromFactory"); + } + + @Override + public Class getObjectType() { + return ExampleBean.class; + } + + @Override + public boolean isSingleton() { + return false; + } + + } + + static class NonspecificFactoryBean implements FactoryBean { + + NonspecificFactoryBean(String value) { + Assert.state(!value.contains("$"), "value should not contain '$'"); + } + + @Override + public ExampleBean getObject() { + return new ExampleBean("fromFactory"); + } + + @Override + public Class getObjectType() { + return ExampleBean.class; + } + + @Override + public boolean isSingleton() { + return false; + } + + } + @TestAnnotation static class ExampleBean { @@ -956,6 +1101,30 @@ public String toString() { } + @TestAnnotation + static class GenericExampleBean { + + private final T value; + + GenericExampleBean(T value) { + this.value = value; + } + + @Override + public String toString() { + return String.valueOf(this.value); + } + + } + + static class CustomGenericExampleBean extends GenericExampleBean { + + CustomGenericExampleBean() { + super("custom subclass"); + } + + } + @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Documented diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnPropertyTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnPropertyTests.java index f663915ce99e..e371d52d1f3a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnPropertyTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnPropertyTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,11 +21,13 @@ import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; import java.util.function.Consumer; +import java.util.stream.Collectors; import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.Test; import org.springframework.boot.WebApplicationType; +import org.springframework.boot.autoconfigure.condition.ConditionEvaluationReport.ConditionAndOutcomes; import org.springframework.boot.builder.SpringApplicationBuilder; import org.springframework.boot.test.util.TestPropertyValues; import org.springframework.context.ConfigurableApplicationContext; @@ -271,6 +273,36 @@ void metaAndDirectAnnotationWithAliasConditionMatchesWhenBothPropertiesAreSet() assertThat(this.context.containsBean("foo")).isTrue(); } + @Test + void multiplePropertiesConditionReportWhenMatched() { + load(MultiplePropertiesRequiredConfiguration.class, "property1=value1", "property2=value2"); + assertThat(this.context.containsBean("foo")).isTrue(); + assertThat(getConditionEvaluationReport()).contains("@ConditionalOnProperty ([property1,property2]) matched"); + } + + @Test + void multiplePropertiesConditionReportWhenDoesNotMatch() { + load(MultiplePropertiesRequiredConfiguration.class, "property1=value1"); + assertThat(getConditionEvaluationReport()) + .contains("@ConditionalOnProperty ([property1,property2]) did not find property 'property2'"); + } + + @Test + void repeatablePropertiesConditionReportWhenMatched() { + load(RepeatablePropertiesRequiredConfiguration.class, "property1=value1", "property2=value2"); + assertThat(this.context.containsBean("foo")).isTrue(); + String report = getConditionEvaluationReport(); + assertThat(report).contains("@ConditionalOnProperty (property1) matched"); + assertThat(report).contains("@ConditionalOnProperty (property2) matched"); + } + + @Test + void repeatablePropertiesConditionReportWhenDoesNotMatch() { + load(RepeatablePropertiesRequiredConfiguration.class, "property1=value1"); + assertThat(getConditionEvaluationReport()) + .contains("@ConditionalOnProperty (property2) did not find property 'property2'"); + } + private void load(Class config, String... environment) { TestPropertyValues.of(environment).applyTo(this.environment); this.context = new SpringApplicationBuilder(config).environment(this.environment) @@ -278,6 +310,16 @@ private void load(Class config, String... environment) { .run(); } + private String getConditionEvaluationReport() { + return ConditionEvaluationReport.get(this.context.getBeanFactory()) + .getConditionAndOutcomesBySource() + .values() + .stream() + .flatMap(ConditionAndOutcomes::stream) + .map(Object::toString) + .collect(Collectors.joining("\n")); + } + @Configuration(proxyBeanMethods = false) @ConditionalOnProperty(name = { "property1", "property2" }) static class MultiplePropertiesRequiredConfiguration { @@ -289,6 +331,18 @@ String foo() { } + @Configuration(proxyBeanMethods = false) + @ConditionalOnProperty("property1") + @ConditionalOnProperty("property2") + static class RepeatablePropertiesRequiredConfiguration { + + @Bean + String foo() { + return "foo"; + } + + } + @Configuration(proxyBeanMethods = false) @ConditionalOnProperty(prefix = "spring.", name = "the-relaxed-property") static class RelaxedPropertiesRequiredConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseAutoConfigurationTests.java index 0480ab955205..dbd10bfffea7 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/couchbase/CouchbaseAutoConfigurationTests.java @@ -197,7 +197,7 @@ void enableSsl() { testClusterEnvironment((env) -> { SecurityConfig securityConfig = env.securityConfig(); assertThat(securityConfig.tlsEnabled()).isTrue(); - assertThat(securityConfig.trustManagerFactory()).isNull(); + assertThat(securityConfig.trustManagerFactory()).isNotNull(); }, "spring.couchbase.env.ssl.enabled=true"); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraDataAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraDataAutoConfigurationTests.java index 932ba62cafce..e6f4f4a82e0c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraDataAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraDataAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,13 +19,13 @@ import java.util.Collections; import com.datastax.oss.driver.api.core.CqlSession; -import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.Test; +import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.cassandra.CassandraAutoConfiguration; import org.springframework.boot.autoconfigure.data.cassandra.city.City; import org.springframework.boot.autoconfigure.domain.EntityScan; -import org.springframework.boot.test.util.TestPropertyValues; +import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.context.annotation.AnnotationConfigApplicationContext; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -33,11 +33,9 @@ import org.springframework.data.cassandra.core.CassandraTemplate; import org.springframework.data.cassandra.core.convert.CassandraConverter; import org.springframework.data.cassandra.core.convert.CassandraCustomConversions; +import org.springframework.data.cassandra.core.cql.CqlTemplate; import org.springframework.data.cassandra.core.mapping.CassandraMappingContext; import org.springframework.data.cassandra.core.mapping.SimpleUserTypeResolver; -import org.springframework.data.domain.ManagedTypes; -import org.springframework.test.util.ReflectionTestUtils; -import org.springframework.util.ObjectUtils; import static org.assertj.core.api.Assertions.assertThat; @@ -50,74 +48,81 @@ */ class CassandraDataAutoConfigurationTests { - private AnnotationConfigApplicationContext context; + private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() + .withPropertyValues("spring.cassandra.keyspaceName=boot_test") + .withUserConfiguration(CassandraMockConfiguration.class) + .withConfiguration( + AutoConfigurations.of(CassandraAutoConfiguration.class, CassandraDataAutoConfiguration.class)); - @AfterEach - void close() { - if (this.context != null) { - this.context.close(); - } + @Test + void cqlTemplateExists() { + this.contextRunner.run((context) -> assertThat(context).hasSingleBean(CqlTemplate.class)); } @Test void templateExists() { - load(CassandraMockConfiguration.class); - assertThat(this.context.getBeanNamesForType(CassandraTemplate.class)).hasSize(1); + this.contextRunner.run((context) -> assertThat(context).hasSingleBean(CassandraTemplate.class)); + } + + @Test + void templateUsesCqlTemplate() { + this.contextRunner.run((context) -> { + assertThat(context).hasSingleBean(CassandraTemplate.class); + assertThat(context.getBean(CassandraTemplate.class).getCqlOperations()) + .isSameAs(context.getBean(CqlTemplate.class)); + }); } @Test void entityScanShouldSetManagedTypes() { - load(EntityScanConfig.class); - CassandraMappingContext mappingContext = this.context.getBean(CassandraMappingContext.class); - ManagedTypes managedTypes = (ManagedTypes) ReflectionTestUtils.getField(mappingContext, "managedTypes"); - assertThat(managedTypes.toList()).containsOnly(City.class); + this.contextRunner.withUserConfiguration(EntityScanConfig.class).run((context) -> { + assertThat(context).hasSingleBean(CassandraMappingContext.class); + CassandraMappingContext mappingContext = context.getBean(CassandraMappingContext.class); + assertThat(mappingContext.getManagedTypes()).singleElement() + .satisfies((typeInformation) -> assertThat(typeInformation.getType()).isEqualTo(City.class)); + }); } @Test void userTypeResolverShouldBeSet() { - load(); - CassandraConverter cassandraConverter = this.context.getBean(CassandraConverter.class); - assertThat(cassandraConverter).extracting("userTypeResolver").isInstanceOf(SimpleUserTypeResolver.class); + this.contextRunner.run((context) -> { + assertThat(context).hasSingleBean(CassandraConverter.class); + assertThat(context.getBean(CassandraConverter.class)).extracting("userTypeResolver") + .isInstanceOf(SimpleUserTypeResolver.class); + }); } @Test void codecRegistryShouldBeSet() { - load(); - CassandraConverter cassandraConverter = this.context.getBean(CassandraConverter.class); - assertThat(cassandraConverter.getCodecRegistry()) - .isSameAs(this.context.getBean(CassandraMockConfiguration.class).codecRegistry); + this.contextRunner.run((context) -> { + assertThat(context).hasSingleBean(CassandraConverter.class); + assertThat(context.getBean(CassandraConverter.class).getCodecRegistry()) + .isSameAs(context.getBean(CassandraMockConfiguration.class).codecRegistry); + }); } @Test void defaultConversions() { - load(); - CassandraTemplate template = this.context.getBean(CassandraTemplate.class); - assertThat(template.getConverter().getConversionService().canConvert(Person.class, String.class)).isFalse(); + this.contextRunner.run((context) -> { + CassandraTemplate template = context.getBean(CassandraTemplate.class); + assertThat(template.getConverter().getConversionService().canConvert(Person.class, String.class)).isFalse(); + }); } @Test void customConversions() { - load(CustomConversionConfig.class); - CassandraTemplate template = this.context.getBean(CassandraTemplate.class); - assertThat(template.getConverter().getConversionService().canConvert(Person.class, String.class)).isTrue(); + this.contextRunner.withUserConfiguration(CustomConversionConfig.class).run((context) -> { + CassandraTemplate template = context.getBean(CassandraTemplate.class); + assertThat(template.getConverter().getConversionService().canConvert(Person.class, String.class)).isTrue(); + }); } @Test void clusterDoesNotExist() { - this.context = new AnnotationConfigApplicationContext(CassandraDataAutoConfiguration.class); - assertThat(this.context.getBeansOfType(CqlSession.class)).isEmpty(); - } - - void load(Class... config) { - AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(); - TestPropertyValues.of("spring.cassandra.keyspaceName:boot_test").applyTo(ctx); - if (!ObjectUtils.isEmpty(config)) { - ctx.register(config); + try (AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext( + CassandraDataAutoConfiguration.class)) { + assertThat(context.getBeansOfType(CqlSession.class)).isEmpty(); } - ctx.register(CassandraMockConfiguration.class, CassandraAutoConfiguration.class, - CassandraDataAutoConfiguration.class); - ctx.refresh(); - this.context = ctx; } @Configuration(proxyBeanMethods = false) diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraReactiveDataAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraReactiveDataAutoConfigurationTests.java index 5e0521c65013..deabd435d53b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraReactiveDataAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraReactiveDataAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,21 +16,19 @@ package org.springframework.boot.autoconfigure.data.cassandra; -import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.Test; +import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.cassandra.CassandraAutoConfiguration; import org.springframework.boot.autoconfigure.data.cassandra.city.City; import org.springframework.boot.autoconfigure.domain.EntityScan; -import org.springframework.boot.test.util.TestPropertyValues; -import org.springframework.context.annotation.AnnotationConfigApplicationContext; +import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.context.annotation.Configuration; import org.springframework.data.cassandra.core.ReactiveCassandraTemplate; import org.springframework.data.cassandra.core.convert.CassandraConverter; +import org.springframework.data.cassandra.core.cql.ReactiveCqlTemplate; import org.springframework.data.cassandra.core.mapping.CassandraMappingContext; import org.springframework.data.cassandra.core.mapping.SimpleUserTypeResolver; -import org.springframework.data.domain.ManagedTypes; -import org.springframework.test.util.ReflectionTestUtils; import static org.assertj.core.api.Assertions.assertThat; @@ -43,50 +41,48 @@ */ class CassandraReactiveDataAutoConfigurationTests { - private AnnotationConfigApplicationContext context; + private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() + .withPropertyValues("spring.cassandra.keyspaceName=boot_test") + .withUserConfiguration(CassandraMockConfiguration.class) + .withConfiguration(AutoConfigurations.of(CassandraAutoConfiguration.class, CassandraDataAutoConfiguration.class, + CassandraReactiveDataAutoConfiguration.class)); - @AfterEach - void close() { - if (this.context != null) { - this.context.close(); - } + @Test + void reactiveCqlTemplateExists() { + this.contextRunner.run((context) -> assertThat(context).hasSingleBean(ReactiveCqlTemplate.class)); } @Test void templateExists() { - load("spring.cassandra.keyspaceName:boot_test"); - assertThat(this.context.getBeanNamesForType(ReactiveCassandraTemplate.class)).hasSize(1); + this.contextRunner.run((context) -> assertThat(context).hasSingleBean(ReactiveCassandraTemplate.class)); } @Test - void entityScanShouldSetManagedTypes() { - load(EntityScanConfig.class, "spring.cassandra.keyspaceName:boot_test"); - CassandraMappingContext mappingContext = this.context.getBean(CassandraMappingContext.class); - ManagedTypes managedTypes = (ManagedTypes) ReflectionTestUtils.getField(mappingContext, "managedTypes"); - assertThat(managedTypes.toList()).containsOnly(City.class); + void templateUsesReactiveCqlTemplate() { + this.contextRunner.run((context) -> { + assertThat(context).hasSingleBean(ReactiveCassandraTemplate.class); + assertThat(context.getBean(ReactiveCassandraTemplate.class).getReactiveCqlOperations()) + .isSameAs(context.getBean(ReactiveCqlTemplate.class)); + }); } @Test - void userTypeResolverShouldBeSet() { - load("spring.cassandra.keyspaceName:boot_test"); - CassandraConverter cassandraConverter = this.context.getBean(CassandraConverter.class); - assertThat(cassandraConverter).extracting("userTypeResolver").isInstanceOf(SimpleUserTypeResolver.class); - } - - private void load(String... environment) { - load(null, environment); + void entityScanShouldSetManagedTypes() { + this.contextRunner.withUserConfiguration(EntityScanConfig.class).run((context) -> { + assertThat(context).hasSingleBean(CassandraMappingContext.class); + CassandraMappingContext mappingContext = context.getBean(CassandraMappingContext.class); + assertThat(mappingContext.getManagedTypes()).singleElement() + .satisfies((typeInformation) -> assertThat(typeInformation.getType()).isEqualTo(City.class)); + }); } - private void load(Class config, String... environment) { - AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(); - TestPropertyValues.of(environment).applyTo(ctx); - if (config != null) { - ctx.register(config); - } - ctx.register(CassandraMockConfiguration.class, CassandraAutoConfiguration.class, - CassandraDataAutoConfiguration.class, CassandraReactiveDataAutoConfiguration.class); - ctx.refresh(); - this.context = ctx; + @Test + void userTypeResolverShouldBeSet() { + this.contextRunner.run((context) -> { + assertThat(context).hasSingleBean(CassandraConverter.class); + assertThat(context.getBean(CassandraConverter.class)).extracting("userTypeResolver") + .isInstanceOf(SimpleUserTypeResolver.class); + }); } @Configuration(proxyBeanMethods = false) diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraRepositoriesAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraRepositoriesAutoConfigurationTests.java index 7e6294e6df4a..236ae1ea6bee 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraRepositoriesAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/cassandra/CassandraRepositoriesAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -79,15 +79,15 @@ void doesNotTriggerDefaultRepositoryDetectionIfCustomized() { @Test void enablingReactiveRepositoriesDisablesImperativeRepositories() { this.contextRunner.withUserConfiguration(DefaultConfiguration.class) - .withPropertyValues("spring.cassandra.repositories.type=reactive") - .run((context) -> assertThat(context).doesNotHaveBean(CityCassandraRepository.class)); + .withPropertyValues("spring.data.cassandra.repositories.type=reactive") + .run((context) -> assertThat(context).doesNotHaveBean(CityRepository.class)); } @Test void enablingNoRepositoriesDisablesImperativeRepositories() { this.contextRunner.withUserConfiguration(DefaultConfiguration.class) - .withPropertyValues("spring.cassandra.repositories.type=none") - .run((context) -> assertThat(context).doesNotHaveBean(CityCassandraRepository.class)); + .withPropertyValues("spring.data.cassandra.repositories.type=none") + .run((context) -> assertThat(context).doesNotHaveBean(CityRepository.class)); } private ManagedTypes getManagedTypes(AssertableApplicationContext context) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/redis/PropertiesRedisConnectionDetailsTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/redis/PropertiesRedisConnectionDetailsTests.java index 4d3aa55ab84f..6b0445440320 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/redis/PropertiesRedisConnectionDetailsTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/redis/PropertiesRedisConnectionDetailsTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,40 +18,54 @@ import java.util.List; +import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; import org.springframework.boot.autoconfigure.data.redis.RedisConnectionDetails.Node; +import org.springframework.boot.ssl.DefaultSslBundleRegistry; +import org.springframework.boot.ssl.SslBundle; import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.Mockito.mock; /** * Tests for {@link PropertiesRedisConnectionDetails}. * * @author Scott Frederick + * @author Moritz Halbritter */ class PropertiesRedisConnectionDetailsTests { - private final RedisProperties properties = new RedisProperties(); + private RedisProperties properties; + + private PropertiesRedisConnectionDetails connectionDetails; + + private DefaultSslBundleRegistry sslBundleRegistry; + + @BeforeEach + void setUp() { + this.properties = new RedisProperties(); + this.sslBundleRegistry = new DefaultSslBundleRegistry(); + this.connectionDetails = new PropertiesRedisConnectionDetails(this.properties, this.sslBundleRegistry); + } @Test void connectionIsConfiguredWithDefaults() { - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); - RedisConnectionDetails.Standalone standalone = connectionDetails.getStandalone(); + RedisConnectionDetails.Standalone standalone = this.connectionDetails.getStandalone(); assertThat(standalone.getHost()).isEqualTo("localhost"); assertThat(standalone.getPort()).isEqualTo(6379); assertThat(standalone.getDatabase()).isEqualTo(0); - assertThat(connectionDetails.getSentinel()).isNull(); - assertThat(connectionDetails.getCluster()).isNull(); - assertThat(connectionDetails.getUsername()).isNull(); - assertThat(connectionDetails.getPassword()).isNull(); + assertThat(this.connectionDetails.getSentinel()).isNull(); + assertThat(this.connectionDetails.getCluster()).isNull(); + assertThat(this.connectionDetails.getUsername()).isNull(); + assertThat(this.connectionDetails.getPassword()).isNull(); } @Test void credentialsAreConfiguredFromUrlWithUsernameAndPassword() { this.properties.setUrl("redis://user:secret@example.com"); - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); - assertThat(connectionDetails.getUsername()).isEqualTo("user"); - assertThat(connectionDetails.getPassword()).isEqualTo("secret"); + assertThat(this.connectionDetails.getUsername()).isEqualTo("user"); + assertThat(this.connectionDetails.getPassword()).isEqualTo("secret"); } @Test @@ -59,9 +73,8 @@ void credentialsAreConfiguredFromUrlWithUsernameAndColon() { this.properties.setUrl("redis://user:@example.com"); this.properties.setUsername("notused"); this.properties.setPassword("notused"); - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); - assertThat(connectionDetails.getUsername()).isEqualTo("user"); - assertThat(connectionDetails.getPassword()).isEmpty(); + assertThat(this.connectionDetails.getUsername()).isEqualTo("user"); + assertThat(this.connectionDetails.getPassword()).isEmpty(); } @Test @@ -69,9 +82,8 @@ void credentialsAreConfiguredFromUrlWithColonAndPassword() { this.properties.setUrl("redis://:secret@example.com"); this.properties.setUsername("notused"); this.properties.setPassword("notused"); - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); - assertThat(connectionDetails.getUsername()).isEmpty(); - assertThat(connectionDetails.getPassword()).isEqualTo("secret"); + assertThat(this.connectionDetails.getUsername()).isEmpty(); + assertThat(this.connectionDetails.getPassword()).isEqualTo("secret"); } @Test @@ -79,18 +91,16 @@ void credentialsAreConfiguredFromUrlWithPasswordOnly() { this.properties.setUrl("redis://secret@example.com"); this.properties.setUsername("notused"); this.properties.setPassword("notused"); - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); - assertThat(connectionDetails.getUsername()).isNull(); - assertThat(connectionDetails.getPassword()).isEqualTo("secret"); + assertThat(this.connectionDetails.getUsername()).isNull(); + assertThat(this.connectionDetails.getPassword()).isEqualTo("secret"); } @Test void credentialsAreConfiguredFromProperties() { this.properties.setUsername("user"); this.properties.setPassword("secret"); - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); - assertThat(connectionDetails.getUsername()).isEqualTo("user"); - assertThat(connectionDetails.getPassword()).isEqualTo("secret"); + assertThat(this.connectionDetails.getUsername()).isEqualTo("user"); + assertThat(this.connectionDetails.getPassword()).isEqualTo("secret"); } @Test @@ -99,11 +109,22 @@ void standaloneIsConfiguredFromUrl() { this.properties.setHost("notused"); this.properties.setPort(9999); this.properties.setDatabase(5); - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); + RedisConnectionDetails.Standalone standalone = this.connectionDetails.getStandalone(); + assertThat(standalone.getHost()).isEqualTo("example.com"); + assertThat(standalone.getPort()).isEqualTo(1234); + assertThat(standalone.getDatabase()).isEqualTo(9999); + } + + @Test + void standaloneIsConfiguredFromUrlWithoutDatabase() { + this.properties.setUrl("redis://example.com:1234"); + this.properties.setDatabase(5); + PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties, + null); RedisConnectionDetails.Standalone standalone = connectionDetails.getStandalone(); assertThat(standalone.getHost()).isEqualTo("example.com"); assertThat(standalone.getPort()).isEqualTo(1234); - assertThat(standalone.getDatabase()).isEqualTo(5); + assertThat(standalone.getDatabase()).isEqualTo(0); } @Test @@ -111,8 +132,7 @@ void standaloneIsConfiguredFromProperties() { this.properties.setHost("example.com"); this.properties.setPort(1234); this.properties.setDatabase(5); - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); - RedisConnectionDetails.Standalone standalone = connectionDetails.getStandalone(); + RedisConnectionDetails.Standalone standalone = this.connectionDetails.getStandalone(); assertThat(standalone.getHost()).isEqualTo("example.com"); assertThat(standalone.getPort()).isEqualTo(1234); assertThat(standalone.getDatabase()).isEqualTo(5); @@ -123,8 +143,7 @@ void clusterIsConfigured() { RedisProperties.Cluster cluster = new RedisProperties.Cluster(); cluster.setNodes(List.of("localhost:1111", "127.0.0.1:2222", "[::1]:3333")); this.properties.setCluster(cluster); - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); - assertThat(connectionDetails.getCluster().getNodes()).containsExactly(new Node("localhost", 1111), + assertThat(this.connectionDetails.getCluster().getNodes()).containsExactly(new Node("localhost", 1111), new Node("127.0.0.1", 2222), new Node("[::1]", 3333)); } @@ -133,9 +152,47 @@ void sentinelIsConfigured() { RedisProperties.Sentinel sentinel = new RedisProperties.Sentinel(); sentinel.setNodes(List.of("localhost:1111", "127.0.0.1:2222", "[::1]:3333")); this.properties.setSentinel(sentinel); - PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties); + this.properties.setDatabase(5); + PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties, + null); assertThat(connectionDetails.getSentinel().getNodes()).containsExactly(new Node("localhost", 1111), new Node("127.0.0.1", 2222), new Node("[::1]", 3333)); + assertThat(connectionDetails.getSentinel().getDatabase()).isEqualTo(5); + } + + @Test + void sentinelDatabaseIsConfiguredFromUrl() { + RedisProperties.Sentinel sentinel = new RedisProperties.Sentinel(); + sentinel.setNodes(List.of("localhost:1111", "127.0.0.1:2222", "[::1]:3333")); + this.properties.setSentinel(sentinel); + this.properties.setUrl("redis://example.com:1234/9999"); + this.properties.setDatabase(5); + PropertiesRedisConnectionDetails connectionDetails = new PropertiesRedisConnectionDetails(this.properties, + null); + assertThat(connectionDetails.getSentinel().getDatabase()).isEqualTo(9999); + } + + @Test + void shouldReturnSslBundle() { + SslBundle bundle1 = mock(SslBundle.class); + this.sslBundleRegistry.registerBundle("bundle-1", bundle1); + this.properties.getSsl().setBundle("bundle-1"); + SslBundle sslBundle = this.connectionDetails.getStandalone().getSslBundle(); + assertThat(sslBundle).isSameAs(bundle1); + } + + @Test + void shouldReturnSystemBundleIfSslIsEnabledButBundleNotSet() { + this.properties.getSsl().setEnabled(true); + SslBundle sslBundle = this.connectionDetails.getStandalone().getSslBundle(); + assertThat(sslBundle).isNotNull(); + } + + @Test + void shouldReturnNullIfSslIsNotEnabled() { + this.properties.getSsl().setEnabled(false); + SslBundle sslBundle = this.connectionDetails.getStandalone().getSslBundle(); + assertThat(sslBundle).isNull(); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/redis/RedisAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/redis/RedisAutoConfigurationTests.java index 2cc868b8687c..f908e6f47d50 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/redis/RedisAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/redis/RedisAutoConfigurationTests.java @@ -19,20 +19,30 @@ import java.time.Duration; import java.util.Arrays; import java.util.EnumSet; +import java.util.Iterator; import java.util.List; import java.util.Set; import java.util.function.Consumer; import java.util.stream.Collectors; +import java.util.stream.Stream; import io.lettuce.core.ClientOptions; +import io.lettuce.core.ReadFrom; +import io.lettuce.core.ReadFrom.Nodes; +import io.lettuce.core.RedisURI; import io.lettuce.core.cluster.ClusterClientOptions; import io.lettuce.core.cluster.ClusterTopologyRefreshOptions.RefreshTrigger; +import io.lettuce.core.cluster.models.partitions.RedisClusterNode; +import io.lettuce.core.models.role.RedisNodeDescription; import io.lettuce.core.resource.DefaultClientResources; import io.lettuce.core.tracing.Tracing; import org.apache.commons.pool2.impl.GenericObjectPoolConfig; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.condition.EnabledForJreRange; import org.junit.jupiter.api.condition.JRE; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.Arguments; +import org.junit.jupiter.params.provider.MethodSource; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.data.redis.RedisProperties.Pool; @@ -113,6 +123,60 @@ void testOverrideRedisConfiguration() { }); } + @ParameterizedTest(name = "{0}") + @MethodSource + void shouldConfigureLettuceReadFromProperty(String type, ReadFrom readFrom) { + this.contextRunner.withPropertyValues("spring.data.redis.lettuce.read-from:" + type).run((context) -> { + LettuceConnectionFactory factory = context.getBean(LettuceConnectionFactory.class); + LettuceClientConfiguration configuration = factory.getClientConfiguration(); + assertThat(configuration.getReadFrom()).hasValue(readFrom); + }); + } + + static Stream shouldConfigureLettuceReadFromProperty() { + return Stream.of(Arguments.of("any", ReadFrom.ANY), Arguments.of("any-replica", ReadFrom.ANY_REPLICA), + Arguments.of("lowest-latency", ReadFrom.LOWEST_LATENCY), Arguments.of("replica", ReadFrom.REPLICA), + Arguments.of("replica-preferred", ReadFrom.REPLICA_PREFERRED), + Arguments.of("upstream", ReadFrom.UPSTREAM), + Arguments.of("upstream-preferred", ReadFrom.UPSTREAM_PREFERRED)); + } + + @Test + void shouldConfigureLettuceRegexReadFromProperty() { + RedisClusterNode node1 = createRedisNode("redis-node-1.region-1.example.com"); + RedisClusterNode node2 = createRedisNode("redis-node-2.region-1.example.com"); + RedisClusterNode node3 = createRedisNode("redis-node-1.region-2.example.com"); + RedisClusterNode node4 = createRedisNode("redis-node-2.region-2.example.com"); + this.contextRunner.withPropertyValues("spring.data.redis.lettuce.read-from:regex:.*region-1.*") + .run((context) -> { + LettuceConnectionFactory factory = context.getBean(LettuceConnectionFactory.class); + LettuceClientConfiguration configuration = factory.getClientConfiguration(); + assertThat(configuration.getReadFrom()).hasValueSatisfying((readFrom) -> { + List result = readFrom.select(new RedisNodes(node1, node2, node3, node4)); + assertThat(result).hasSize(2).containsExactly(node1, node2); + }); + }); + } + + @Test + void shouldConfigureLettuceSubnetReadFromProperty() { + RedisClusterNode nodeInSubnetIpv4 = createRedisNode("192.0.2.1"); + RedisClusterNode nodeNotInSubnetIpv4 = createRedisNode("198.51.100.1"); + RedisClusterNode nodeInSubnetIpv6 = createRedisNode("2001:db8:abcd:0000::1"); + RedisClusterNode nodeNotInSubnetIpv6 = createRedisNode("2001:db8:abcd:1000::"); + this.contextRunner + .withPropertyValues("spring.data.redis.lettuce.read-from:subnet:192.0.2.0/24,2001:db8:abcd:0000::/52") + .run((context) -> { + LettuceConnectionFactory factory = context.getBean(LettuceConnectionFactory.class); + LettuceClientConfiguration configuration = factory.getClientConfiguration(); + assertThat(configuration.getReadFrom()).hasValueSatisfying((readFrom) -> { + List result = readFrom.select(new RedisNodes(nodeInSubnetIpv4, + nodeNotInSubnetIpv4, nodeInSubnetIpv6, nodeNotInSubnetIpv6)); + assertThat(result).hasSize(2).containsExactly(nodeInSubnetIpv4, nodeInSubnetIpv6); + }); + }); + } + @Test void testCustomizeClientResources() { Tracing tracing = mock(Tracing.class); @@ -634,6 +698,32 @@ private String getUserName(LettuceConnectionFactory factory) { return ReflectionTestUtils.invokeMethod(factory, "getRedisUsername"); } + private RedisClusterNode createRedisNode(String host) { + RedisClusterNode node = new RedisClusterNode(); + node.setUri(RedisURI.Builder.redis(host).build()); + return node; + } + + private static final class RedisNodes implements Nodes { + + private final List descriptions; + + RedisNodes(RedisNodeDescription... descriptions) { + this.descriptions = List.of(descriptions); + } + + @Override + public List getNodes() { + return this.descriptions; + } + + @Override + public Iterator iterator() { + return this.descriptions.iterator(); + } + + } + @Configuration(proxyBeanMethods = false) static class CustomConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/jpa/JpaWebAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/web/SpringDataWebAutoConfigurationJpaTests.java similarity index 91% rename from spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/jpa/JpaWebAutoConfigurationTests.java rename to spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/web/SpringDataWebAutoConfigurationJpaTests.java index cc5acd7c24dd..433c40269a7c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/jpa/JpaWebAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/data/web/SpringDataWebAutoConfigurationJpaTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -14,15 +14,15 @@ * limitations under the License. */ -package org.springframework.boot.autoconfigure.data.jpa; +package org.springframework.boot.autoconfigure.data.web; import org.junit.jupiter.api.Test; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.TestAutoConfigurationPackage; +import org.springframework.boot.autoconfigure.data.jpa.JpaRepositoriesAutoConfiguration; import org.springframework.boot.autoconfigure.data.jpa.city.City; import org.springframework.boot.autoconfigure.data.jpa.city.CityRepository; -import org.springframework.boot.autoconfigure.data.web.SpringDataWebAutoConfiguration; import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; import org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration; import org.springframework.boot.test.context.runner.WebApplicationContextRunner; @@ -42,7 +42,7 @@ * @author Dave Syer * @author Stephane Nicoll */ -class JpaWebAutoConfigurationTests { +class SpringDataWebAutoConfigurationJpaTests { private final WebApplicationContextRunner contextRunner = new WebApplicationContextRunner() .withConfiguration(AutoConfigurations.of(DataSourceAutoConfiguration.class, HibernateJpaAutoConfiguration.class, diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/domain/EntityScanPackagesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/domain/EntityScanPackagesTests.java index 0289be2ee956..ce789779fdb7 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/domain/EntityScanPackagesTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/domain/EntityScanPackagesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -69,7 +69,7 @@ void getShouldReturnRegisterPackages() { @Test void registerFromArrayWhenRegistryIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> EntityScanPackages.register(null)) - .withMessageContaining("Registry must not be null"); + .withMessageContaining("'registry' must not be null"); } @@ -78,14 +78,14 @@ void registerFromArrayWhenPackageNamesIsNullShouldThrowException() { this.context = new AnnotationConfigApplicationContext(); assertThatIllegalArgumentException() .isThrownBy(() -> EntityScanPackages.register(this.context, (String[]) null)) - .withMessageContaining("PackageNames must not be null"); + .withMessageContaining("'packageNames' must not be null"); } @Test void registerFromCollectionWhenRegistryIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> EntityScanPackages.register(null, Collections.emptyList())) - .withMessageContaining("Registry must not be null"); + .withMessageContaining("'registry' must not be null"); } @Test @@ -93,7 +93,7 @@ void registerFromCollectionWhenPackageNamesIsNullShouldThrowException() { this.context = new AnnotationConfigApplicationContext(); assertThatIllegalArgumentException() .isThrownBy(() -> EntityScanPackages.register(this.context, (Collection) null)) - .withMessageContaining("PackageNames must not be null"); + .withMessageContaining("'packageNames' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/domain/EntityScannerTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/domain/EntityScannerTests.java index ec707d9c8382..6c3a99c47b6b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/domain/EntityScannerTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/domain/EntityScannerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,7 +53,7 @@ class EntityScannerTests { @Test void createWhenContextIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new EntityScanner(null)) - .withMessageContaining("Context must not be null"); + .withMessageContaining("'context' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/GraphQlAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/GraphQlAutoConfigurationTests.java index be1efca974c1..7fe6c2ac94f4 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/GraphQlAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/GraphQlAutoConfigurationTests.java @@ -35,9 +35,7 @@ import org.springframework.aot.hint.predicate.RuntimeHintsPredicates; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.graphql.GraphQlAutoConfiguration.GraphQlResourcesRuntimeHints; -import org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListener; import org.springframework.boot.autoconfigure.task.TaskExecutionAutoConfiguration; -import org.springframework.boot.logging.LogLevel; import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.boot.test.system.CapturedOutput; import org.springframework.boot.test.system.OutputCaptureExtension; @@ -89,12 +87,11 @@ class GraphQlAutoConfigurationTests { @Test void shouldContributeDefaultBeans() { - this.contextRunner.withInitializer(ConditionEvaluationReportLoggingListener.forLogLevel(LogLevel.INFO)) - .run((context) -> assertThat(context).hasSingleBean(GraphQlSource.class) - .hasSingleBean(BatchLoaderRegistry.class) - .hasSingleBean(ExecutionGraphQlService.class) - .hasSingleBean(AnnotatedControllerConfigurer.class) - .hasSingleBean(EncodingCursorStrategy.class)); + this.contextRunner.run((context) -> assertThat(context).hasSingleBean(GraphQlSource.class) + .hasSingleBean(BatchLoaderRegistry.class) + .hasSingleBean(ExecutionGraphQlService.class) + .hasSingleBean(AnnotatedControllerConfigurer.class) + .hasSingleBean(EncodingCursorStrategy.class)); } @Test @@ -140,10 +137,33 @@ void shouldScanLocationsWithCustomExtension() { } @Test - void shouldBackOffWithCustomGraphQlSource() { + @WithResource(name = "graphql/types/person.custom", content = """ + type Person { + id: ID + name: String + } + """) + void shouldConfigureAdditionalSchemaFiles() { + this.contextRunner + .withPropertyValues("spring.graphql.schema.additional-files=classpath:graphql/types/person.custom") + .run((context) -> { + assertThat(context).hasSingleBean(GraphQlSource.class); + GraphQlSource graphQlSource = context.getBean(GraphQlSource.class); + GraphQLSchema schema = graphQlSource.schema(); + assertThat(schema.getObjectType("Book")).isNotNull(); + assertThat(schema.getObjectType("Person")).isNotNull(); + }); + } + + @Test + void shouldUseCustomGraphQlSource() { this.contextRunner.withUserConfiguration(CustomGraphQlSourceConfiguration.class).run((context) -> { assertThat(context).getBeanNames(GraphQlSource.class).containsOnly("customGraphQlSource"); - assertThat(context).hasSingleBean(GraphQlProperties.class); + assertThat(context).hasSingleBean(GraphQlProperties.class) + .hasSingleBean(BatchLoaderRegistry.class) + .hasSingleBean(ExecutionGraphQlService.class) + .hasSingleBean(AnnotatedControllerConfigurer.class) + .hasSingleBean(EncodingCursorStrategy.class); }); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/reactive/GraphQlWebFluxAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/reactive/GraphQlWebFluxAutoConfigurationTests.java index 2c15f2b1249c..a07d11288079 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/reactive/GraphQlWebFluxAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/reactive/GraphQlWebFluxAutoConfigurationTests.java @@ -42,6 +42,7 @@ import org.springframework.graphql.server.WebGraphQlHandler; import org.springframework.graphql.server.WebGraphQlInterceptor; import org.springframework.graphql.server.webflux.GraphQlHttpHandler; +import org.springframework.graphql.server.webflux.GraphQlSseHandler; import org.springframework.graphql.server.webflux.GraphQlWebSocketHandler; import org.springframework.http.HttpHeaders; import org.springframework.http.HttpStatus; @@ -271,6 +272,24 @@ void shouldConfigureWebSocketProperties() { }); } + @Test + void shouldConfigureSseTimeout() { + this.contextRunner.withPropertyValues("spring.graphql.http.sse.timeout=10s").run((context) -> { + assertThat(context).hasSingleBean(GraphQlSseHandler.class); + GraphQlSseHandler handler = context.getBean(GraphQlSseHandler.class); + assertThat(handler).hasFieldOrPropertyWithValue("timeout", Duration.ofSeconds(10)); + }); + } + + @Test + void shouldConfigureSseKeepAlive() { + this.contextRunner.withPropertyValues("spring.graphql.http.sse.keep-alive=5s").run((context) -> { + assertThat(context).hasSingleBean(GraphQlSseHandler.class); + GraphQlSseHandler handler = context.getBean(GraphQlSseHandler.class); + assertThat(handler).hasFieldOrPropertyWithValue("keepAliveDuration", Duration.ofSeconds(5)); + }); + } + @Test void routerFunctionShouldHaveOrderZero() { this.contextRunner.withUserConfiguration(CustomRouterFunctions.class).run((context) -> { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/servlet/GraphQlWebMvcAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/servlet/GraphQlWebMvcAutoConfigurationTests.java index 67de43459426..e8b9bb8f8ee0 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/servlet/GraphQlWebMvcAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/graphql/servlet/GraphQlWebMvcAutoConfigurationTests.java @@ -101,13 +101,22 @@ void shouldContributeDefaultBeans() { @Test void shouldConfigureSseTimeout() { - this.contextRunner.withPropertyValues("spring.graphql.sse.timeout=10s").run((context) -> { + this.contextRunner.withPropertyValues("spring.graphql.http.sse.timeout=10s").run((context) -> { assertThat(context).hasSingleBean(GraphQlSseHandler.class); GraphQlSseHandler handler = context.getBean(GraphQlSseHandler.class); assertThat(handler).hasFieldOrPropertyWithValue("timeout", Duration.ofSeconds(10)); }); } + @Test + void shouldConfigureSseKeepAlive() { + this.contextRunner.withPropertyValues("spring.graphql.http.sse.keep-alive=5s").run((context) -> { + assertThat(context).hasSingleBean(GraphQlSseHandler.class); + GraphQlSseHandler handler = context.getBean(GraphQlSseHandler.class); + assertThat(handler).hasFieldOrPropertyWithValue("keepAliveDuration", Duration.ofSeconds(5)); + }); + } + @Test void simpleQueryShouldWork() { withMockMvc((mvc) -> { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/groovy/template/GroovyTemplateAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/groovy/template/GroovyTemplateAutoConfigurationTests.java index d2eba9d5fb01..885b950f462d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/groovy/template/GroovyTemplateAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/groovy/template/GroovyTemplateAutoConfigurationTests.java @@ -22,8 +22,11 @@ import java.util.Collections; import java.util.HashMap; import java.util.Locale; +import java.util.Map; +import groovy.text.markup.BaseTemplate; import groovy.text.markup.MarkupTemplateEngine; +import groovy.text.markup.TemplateConfiguration; import jakarta.servlet.http.HttpServletRequest; import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.BeforeEach; @@ -185,11 +188,73 @@ void renderTemplate() throws Exception { } @Test + @Deprecated(since = "3.5.0", forRemoval = true) void customConfiguration() { registerAndRefreshContext("spring.groovy.template.configuration.auto-indent:true"); assertThat(this.context.getBean(GroovyMarkupConfigurer.class).isAutoIndent()).isTrue(); } + @Test + void enableAutoEscape() { + registerAndRefreshContext("spring.groovy.template.auto-escape:true"); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).isAutoEscape()).isTrue(); + } + + @Test + void enableAutoIndent() { + registerAndRefreshContext("spring.groovy.template.auto-indent:true"); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).isAutoIndent()).isTrue(); + } + + @Test + void customAutoIndentString() { + registerAndRefreshContext("spring.groovy.template.auto-indent-string:\\t"); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).getAutoIndentString()).isEqualTo("\\t"); + } + + @Test + void enableAutoNewLine() { + registerAndRefreshContext("spring.groovy.template.auto-new-line:true"); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).isAutoNewLine()).isTrue(); + } + + @Test + void customBaseTemplateClass() { + registerAndRefreshContext("spring.groovy.template.base-template-class:" + CustomBaseTemplate.class.getName()); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).getBaseTemplateClass()) + .isEqualTo(CustomBaseTemplate.class); + } + + @Test + void customDeclarationEncoding() { + registerAndRefreshContext("spring.groovy.template.declaration-encoding:UTF-8"); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).getDeclarationEncoding()).isEqualTo("UTF-8"); + } + + @Test + void enableExpandEmptyElements() { + registerAndRefreshContext("spring.groovy.template.expand-empty-elements:true"); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).isExpandEmptyElements()).isTrue(); + } + + @Test + void customLocale() { + registerAndRefreshContext("spring.groovy.template.locale:en_US"); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).getLocale()).isEqualTo(Locale.US); + } + + @Test + void customNewLineString() { + registerAndRefreshContext("spring.groovy.template.new-line-string:\\r\\n"); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).getNewLineString()).isEqualTo("\\r\\n"); + } + + @Test + void enableUseDoubleQuotes() { + registerAndRefreshContext("spring.groovy.template.use-double-quotes:true"); + assertThat(this.context.getBean(GroovyMarkupConfigurer.class).isUseDoubleQuotes()).isTrue(); + } + private void registerAndRefreshContext(String... env) { TestPropertyValues.of(env).applyTo(this.context); this.context.register(GroovyTemplateAutoConfiguration.class); @@ -212,4 +277,19 @@ private MockHttpServletResponse render(String viewName, Locale locale) throws Ex return response; } + static class CustomBaseTemplate extends BaseTemplate { + + @SuppressWarnings("rawtypes") + CustomBaseTemplate(MarkupTemplateEngine templateEngine, Map model, Map modelTypes, + TemplateConfiguration configuration) { + super(templateEngine, model, modelTypes, configuration); + } + + @Override + public Object run() { + return null; + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/h2/H2ConsolePropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/h2/H2ConsolePropertiesTests.java index e60e6abcfa79..bebfd11bf912 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/h2/H2ConsolePropertiesTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/h2/H2ConsolePropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,21 +31,21 @@ class H2ConsolePropertiesTests { void pathMustNotBeEmpty() { H2ConsoleProperties properties = new H2ConsoleProperties(); assertThatIllegalArgumentException().isThrownBy(() -> properties.setPath("")) - .withMessageContaining("Path must have length greater than 1"); + .withMessageContaining("'path' must have length greater than 1"); } @Test void pathMustHaveLengthGreaterThanOne() { H2ConsoleProperties properties = new H2ConsoleProperties(); assertThatIllegalArgumentException().isThrownBy(() -> properties.setPath("/")) - .withMessageContaining("Path must have length greater than 1"); + .withMessageContaining("'path' must have length greater than 1"); } @Test void customPathMustBeginWithASlash() { H2ConsoleProperties properties = new H2ConsoleProperties(); assertThatIllegalArgumentException().isThrownBy(() -> properties.setPath("custom")) - .withMessageContaining("Path must start with '/'"); + .withMessageContaining("'path' must start with '/'"); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastAutoConfigurationServerTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastAutoConfigurationServerTests.java index 4d5e6c478c82..682286c1f26d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastAutoConfigurationServerTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/hazelcast/HazelcastAutoConfigurationServerTests.java @@ -41,7 +41,6 @@ import org.springframework.boot.test.context.assertj.AssertableApplicationContext; import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.boot.test.context.runner.ContextConsumer; -import org.springframework.boot.testsupport.classpath.ClassPathExclusions; import org.springframework.boot.testsupport.classpath.resources.WithResource; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -55,7 +54,6 @@ * * @author Stephane Nicoll */ -@ClassPathExclusions("hazelcast-client-*.jar") class HazelcastAutoConfigurationServerTests { private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/HttpMessageConvertersAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/HttpMessageConvertersAutoConfigurationTests.java index bf760a5719a7..85bbd33f4fbb 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/HttpMessageConvertersAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/HttpMessageConvertersAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -130,6 +130,17 @@ void gsonCustomConverter() { @Test void gsonCanBePreferred() { + allOptionsRunner().withPropertyValues("spring.http.converters.preferred-json-mapper:gson").run((context) -> { + assertConverterBeanExists(context, GsonHttpMessageConverter.class, "gsonHttpMessageConverter"); + assertConverterBeanRegisteredWithHttpMessageConverters(context, GsonHttpMessageConverter.class); + assertThat(context).doesNotHaveBean(JsonbHttpMessageConverter.class); + assertThat(context).doesNotHaveBean(MappingJackson2HttpMessageConverter.class); + }); + } + + @Test + @Deprecated(since = "3.5.0", forRemoval = true) + void gsonCanBePreferredWithDeprecatedProperty() { allOptionsRunner().withPropertyValues("spring.mvc.converters.preferred-json-mapper:gson").run((context) -> { assertConverterBeanExists(context, GsonHttpMessageConverter.class, "gsonHttpMessageConverter"); assertConverterBeanRegisteredWithHttpMessageConverters(context, GsonHttpMessageConverter.class); @@ -138,6 +149,20 @@ void gsonCanBePreferred() { }); } + @Test + @Deprecated(since = "3.5.0", forRemoval = true) + void gsonCanBePreferredWithNonDeprecatedPropertyTakingPrecedence() { + allOptionsRunner() + .withPropertyValues("spring.http.converters.preferred-json-mapper:gson", + "spring.mvc.converters.preferred-json-mapper:jackson") + .run((context) -> { + assertConverterBeanExists(context, GsonHttpMessageConverter.class, "gsonHttpMessageConverter"); + assertConverterBeanRegisteredWithHttpMessageConverters(context, GsonHttpMessageConverter.class); + assertThat(context).doesNotHaveBean(JsonbHttpMessageConverter.class); + assertThat(context).doesNotHaveBean(MappingJackson2HttpMessageConverter.class); + }); + } + @Test void jsonbNotAvailable() { this.contextRunner.run((context) -> { @@ -161,6 +186,17 @@ void jsonbCustomConverter() { @Test void jsonbCanBePreferred() { + allOptionsRunner().withPropertyValues("spring.http.converters.preferred-json-mapper:jsonb").run((context) -> { + assertConverterBeanExists(context, JsonbHttpMessageConverter.class, "jsonbHttpMessageConverter"); + assertConverterBeanRegisteredWithHttpMessageConverters(context, JsonbHttpMessageConverter.class); + assertThat(context).doesNotHaveBean(GsonHttpMessageConverter.class); + assertThat(context).doesNotHaveBean(MappingJackson2HttpMessageConverter.class); + }); + } + + @Test + @Deprecated(since = "3.5.0", forRemoval = true) + void jsonbCanBePreferredWithDeprecatedProperty() { allOptionsRunner().withPropertyValues("spring.mvc.converters.preferred-json-mapper:jsonb").run((context) -> { assertConverterBeanExists(context, JsonbHttpMessageConverter.class, "jsonbHttpMessageConverter"); assertConverterBeanRegisteredWithHttpMessageConverters(context, JsonbHttpMessageConverter.class); @@ -169,6 +205,20 @@ void jsonbCanBePreferred() { }); } + @Test + @Deprecated(since = "3.5.0", forRemoval = true) + void jsonbCanBePreferredWithNonDeprecatedPropertyTakingPrecedence() { + allOptionsRunner() + .withPropertyValues("spring.http.converters.preferred-json-mapper:jsonb", + "spring.mvc.converters.preferred-json-mapper:gson") + .run((context) -> { + assertConverterBeanExists(context, JsonbHttpMessageConverter.class, "jsonbHttpMessageConverter"); + assertConverterBeanRegisteredWithHttpMessageConverters(context, JsonbHttpMessageConverter.class); + assertThat(context).doesNotHaveBean(GsonHttpMessageConverter.class); + assertThat(context).doesNotHaveBean(MappingJackson2HttpMessageConverter.class); + }); + } + @Test void stringDefaultConverter() { this.contextRunner.run(assertConverter(StringHttpMessageConverter.class, "stringHttpMessageConverter")); @@ -285,10 +335,10 @@ void shouldRegisterHints() { RuntimeHints hints = new RuntimeHints(); new HttpMessageConvertersAutoConfigurationRuntimeHints().registerHints(hints, getClass().getClassLoader()); assertThat(RuntimeHintsPredicates.reflection().onType(Encoding.class)).accepts(hints); - assertThat(RuntimeHintsPredicates.reflection().onMethod(Encoding.class, "getCharset")).accepts(hints); - assertThat(RuntimeHintsPredicates.reflection().onMethod(Encoding.class, "setCharset")).accepts(hints); - assertThat(RuntimeHintsPredicates.reflection().onMethod(Encoding.class, "isForce")).accepts(hints); - assertThat(RuntimeHintsPredicates.reflection().onMethod(Encoding.class, "setForce")).accepts(hints); + assertThat(RuntimeHintsPredicates.reflection().onMethod(Encoding.class, "getCharset").invoke()).accepts(hints); + assertThat(RuntimeHintsPredicates.reflection().onMethod(Encoding.class, "setCharset").invoke()).accepts(hints); + assertThat(RuntimeHintsPredicates.reflection().onMethod(Encoding.class, "isForce").invoke()).accepts(hints); + assertThat(RuntimeHintsPredicates.reflection().onMethod(Encoding.class, "setForce").invoke()).accepts(hints); assertThat(RuntimeHintsPredicates.reflection().onMethod(Encoding.class, "shouldForce")).rejects(hints); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/HttpClientAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/HttpClientAutoConfigurationTests.java index 85a23fae4a02..7f2ef917ad96 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/HttpClientAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/HttpClientAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,9 +27,14 @@ import org.springframework.boot.http.client.ClientHttpRequestFactoryBuilder; import org.springframework.boot.http.client.ClientHttpRequestFactorySettings; import org.springframework.boot.http.client.ClientHttpRequestFactorySettings.Redirects; +import org.springframework.boot.http.client.HttpComponentsClientHttpRequestFactoryBuilder; +import org.springframework.boot.http.client.JettyClientHttpRequestFactoryBuilder; import org.springframework.boot.http.client.SimpleClientHttpRequestFactoryBuilder; import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.http.client.ClientHttpRequestFactory; import static org.assertj.core.api.Assertions.assertThat; @@ -89,4 +94,30 @@ void whenReactiveWebApplicationBeansAreNotConfigured() { .doesNotHaveBean(ClientHttpRequestFactorySettings.class)); } + @Test + void clientHttpRequestFactoryBuilderCustomizersAreApplied() { + this.contextRunner.withUserConfiguration(ClientHttpRequestFactoryBuilderCustomizersConfiguration.class) + .run((context) -> { + ClientHttpRequestFactory factory = context.getBean(ClientHttpRequestFactoryBuilder.class).build(); + assertThat(factory).extracting("connectTimeout").isEqualTo(5L); + }); + } + + @Configuration(proxyBeanMethods = false) + static class ClientHttpRequestFactoryBuilderCustomizersConfiguration { + + @Bean + ClientHttpRequestFactoryBuilderCustomizer httpComponentsCustomizer() { + return (builder) -> builder.withCustomizer((factory) -> factory.setConnectTimeout(5)); + } + + @Bean + ClientHttpRequestFactoryBuilderCustomizer jettyCustomizer() { + return (builder) -> { + throw new IllegalStateException(); + }; + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/HttpClientPropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/HttpClientPropertiesTests.java index 16d66577ab72..43b4b7c9ec02 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/HttpClientPropertiesTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/HttpClientPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,7 +19,7 @@ import org.junit.jupiter.api.Nested; import org.junit.jupiter.api.Test; -import org.springframework.boot.autoconfigure.http.client.HttpClientProperties.Factory; +import org.springframework.boot.autoconfigure.http.client.AbstractHttpRequestFactoryProperties.Factory; import org.springframework.boot.http.client.HttpComponentsClientHttpRequestFactoryBuilder; import org.springframework.boot.http.client.JdkClientHttpRequestFactoryBuilder; import org.springframework.boot.http.client.JettyClientHttpRequestFactoryBuilder; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/reactive/ClientHttpConnectorAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/reactive/ClientHttpConnectorAutoConfigurationTests.java new file mode 100644 index 000000000000..efe002406804 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/reactive/ClientHttpConnectorAutoConfigurationTests.java @@ -0,0 +1,204 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.client.reactive; + +import java.time.Duration; +import java.util.ArrayList; +import java.util.List; + +import org.apache.hc.client5.http.impl.async.HttpAsyncClients; +import org.junit.jupiter.api.Test; +import reactor.netty.http.client.HttpClient; + +import org.springframework.beans.factory.config.BeanDefinition; +import org.springframework.boot.autoconfigure.AutoConfigurations; +import org.springframework.boot.autoconfigure.ssl.SslAutoConfiguration; +import org.springframework.boot.http.client.HttpRedirects; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.ClientHttpConnectorSettings; +import org.springframework.boot.http.client.reactive.JdkClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.JettyClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.ReactorClientHttpConnectorBuilder; +import org.springframework.boot.test.context.FilteredClassLoader; +import org.springframework.boot.test.context.runner.ApplicationContextRunner; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.http.client.ReactorResourceFactory; +import org.springframework.http.client.reactive.ClientHttpConnector; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.Mockito.mock; + +/** + * Tests for {@link ClientHttpConnectorAutoConfiguration} + * + * @author Brian Clozel + * @author Phillip Webb + */ +class ClientHttpConnectorAutoConfigurationTests { + + private final ApplicationContextRunner contextRunner = new ApplicationContextRunner().withConfiguration( + AutoConfigurations.of(ClientHttpConnectorAutoConfiguration.class, SslAutoConfiguration.class)); + + @Test + void whenReactorIsAvailableThenReactorBeansAreDefined() { + this.contextRunner.run((context) -> { + BeanDefinition connectorDefinition = context.getBeanFactory().getBeanDefinition("clientHttpConnector"); + assertThat(connectorDefinition.isLazyInit()).isTrue(); + assertThat(context).hasSingleBean(ReactorResourceFactory.class); + assertThat(context.getBean(ClientHttpConnectorBuilder.class)) + .isExactlyInstanceOf(ReactorClientHttpConnectorBuilder.class); + }); + } + + @Test + void whenReactorIsUnavailableThenJettyClientBeansAreDefined() { + this.contextRunner.withClassLoader(new FilteredClassLoader(HttpClient.class)).run((context) -> { + BeanDefinition connectorDefinition = context.getBeanFactory().getBeanDefinition("clientHttpConnector"); + assertThat(connectorDefinition.isLazyInit()).isTrue(); + assertThat(context.getBean(ClientHttpConnectorBuilder.class)) + .isExactlyInstanceOf(JettyClientHttpConnectorBuilder.class); + }); + } + + @Test + void whenReactorAndHttpClientAreUnavailableThenJettyClientBeansAreDefined() { + this.contextRunner.withClassLoader(new FilteredClassLoader(HttpClient.class, HttpAsyncClients.class)) + .run((context) -> { + BeanDefinition connectorDefinition = context.getBeanFactory().getBeanDefinition("clientHttpConnector"); + assertThat(connectorDefinition.isLazyInit()).isTrue(); + assertThat(context.getBean(ClientHttpConnectorBuilder.class)) + .isExactlyInstanceOf(JettyClientHttpConnectorBuilder.class); + }); + } + + @Test + void whenReactorAndHttpClientAndJettyAreUnavailableThenJdkClientBeansAreDefined() { + this.contextRunner + .withClassLoader(new FilteredClassLoader(HttpClient.class, HttpAsyncClients.class, + org.eclipse.jetty.client.HttpClient.class)) + .run((context) -> { + BeanDefinition connectorDefinition = context.getBeanFactory().getBeanDefinition("clientHttpConnector"); + assertThat(connectorDefinition.isLazyInit()).isTrue(); + assertThat(context.getBean(ClientHttpConnectorBuilder.class)) + .isExactlyInstanceOf(JdkClientHttpConnectorBuilder.class); + }); + } + + @Test + void shouldNotOverrideCustomClientConnector() { + this.contextRunner.withUserConfiguration(CustomClientHttpConnectorConfig.class).run((context) -> { + assertThat(context).hasSingleBean(ClientHttpConnector.class); + assertThat(context).hasBean("customConnector"); + }); + } + + @Test + void shouldUseCustomReactorResourceFactory() { + this.contextRunner.withUserConfiguration(CustomReactorResourceConfig.class).run((context) -> { + assertThat(context).hasSingleBean(ClientHttpConnector.class); + assertThat(context).hasSingleBean(ReactorResourceFactory.class); + assertThat(context).hasBean("customReactorResourceFactory"); + }); + } + + @Test + void configuresDetectedClientHttpConnectorBuilderBuilder() { + this.contextRunner.run((context) -> assertThat(context).hasSingleBean(ClientHttpConnectorBuilder.class)); + } + + @Test + void configuresDefinedClientHttpConnectorBuilder() { + this.contextRunner.withPropertyValues("spring.http.reactiveclient.settings.connector=jetty") + .run((context) -> assertThat(context.getBean(ClientHttpConnectorBuilder.class)) + .isInstanceOf(JettyClientHttpConnectorBuilder.class)); + } + + @Test + void configuresClientHttpConnectorSettings() { + this.contextRunner.withPropertyValues(sslPropertyValues().toArray(String[]::new)) + .withPropertyValues("spring.http.reactiveclient.settings.redirects=dont-follow", + "spring.http.reactiveclient.settings.connect-timeout=10s", + "spring.http.reactiveclient.settings.read-timeout=20s", + "spring.http.reactiveclient.settings.ssl.bundle=test") + .run((context) -> { + ClientHttpConnectorSettings settings = context.getBean(ClientHttpConnectorSettings.class); + assertThat(settings.redirects()).isEqualTo(HttpRedirects.DONT_FOLLOW); + assertThat(settings.connectTimeout()).isEqualTo(Duration.ofSeconds(10)); + assertThat(settings.readTimeout()).isEqualTo(Duration.ofSeconds(20)); + assertThat(settings.sslBundle().getKey().getAlias()).isEqualTo("alias1"); + }); + } + + private List sslPropertyValues() { + List propertyValues = new ArrayList<>(); + String location = "classpath:org/springframework/boot/autoconfigure/ssl/"; + propertyValues.add("spring.ssl.bundle.pem.test.key.alias=alias1"); + propertyValues.add("spring.ssl.bundle.pem.test.truststore.type=PKCS12"); + propertyValues.add("spring.ssl.bundle.pem.test.truststore.certificate=" + location + "rsa-cert.pem"); + propertyValues.add("spring.ssl.bundle.pem.test.truststore.private-key=" + location + "rsa-key.pem"); + return propertyValues; + } + + @Test + void clientHttpConnectorBuilderCustomizersAreApplied() { + this.contextRunner.withPropertyValues("spring.http.reactiveclient.settings.connector=jdk") + .withUserConfiguration(ClientHttpConnectorBuilderCustomizersConfiguration.class) + .run((context) -> { + ClientHttpConnector connector = context.getBean(ClientHttpConnectorBuilder.class).build(); + assertThat(connector).extracting("readTimeout").isEqualTo(Duration.ofSeconds(5)); + }); + } + + @Configuration(proxyBeanMethods = false) + static class CustomClientHttpConnectorConfig { + + @Bean + ClientHttpConnector customConnector() { + return mock(ClientHttpConnector.class); + } + + } + + @Configuration(proxyBeanMethods = false) + static class CustomReactorResourceConfig { + + @Bean + ReactorResourceFactory customReactorResourceFactory() { + return new ReactorResourceFactory(); + } + + } + + @Configuration(proxyBeanMethods = false) + static class ClientHttpConnectorBuilderCustomizersConfiguration { + + @Bean + ClientHttpConnectorBuilderCustomizer jdkCustomizer() { + return (builder) -> builder.withCustomizer((connector) -> connector.setReadTimeout(Duration.ofSeconds(5))); + } + + @Bean + ClientHttpConnectorBuilderCustomizer jettyCustomizer() { + return (builder) -> { + throw new IllegalStateException(); + }; + } + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/reactive/HttpReactiveClientSettingsPropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/reactive/HttpReactiveClientSettingsPropertiesTests.java new file mode 100644 index 000000000000..538965262c72 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/client/reactive/HttpReactiveClientSettingsPropertiesTests.java @@ -0,0 +1,63 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.http.client.reactive; + +import org.junit.jupiter.api.Nested; +import org.junit.jupiter.api.Test; + +import org.springframework.boot.autoconfigure.http.client.reactive.AbstractClientHttpConnectorProperties.Connector; +import org.springframework.boot.http.client.reactive.HttpComponentsClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.JdkClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.JettyClientHttpConnectorBuilder; +import org.springframework.boot.http.client.reactive.ReactorClientHttpConnectorBuilder; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link HttpReactiveClientSettingsProperties}. + * + * @author Phillip Webb + */ +class HttpReactiveClientSettingsPropertiesTests { + + @Nested + class ConnectorTests { + + @Test + void reactorBuilder() { + assertThat(Connector.REACTOR.builder()).isInstanceOf(ReactorClientHttpConnectorBuilder.class); + } + + @Test + void jettyBuilder() { + assertThat(Connector.JETTY.builder()).isInstanceOf(JettyClientHttpConnectorBuilder.class); + } + + @Test + void httpComponentsBuilder() { + assertThat(Connector.HTTP_COMPONENTS.builder()) + .isInstanceOf(HttpComponentsClientHttpConnectorBuilder.class); + } + + @Test + void jdkBuilder() { + assertThat(Connector.JDK.builder()).isInstanceOf(JdkClientHttpConnectorBuilder.class); + } + + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/codec/CodecsAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/codec/CodecsAutoConfigurationTests.java index a32ee3da12e7..21364bd062dd 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/codec/CodecsAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/http/codec/CodecsAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,22 +16,21 @@ package org.springframework.boot.autoconfigure.http.codec; -import java.lang.reflect.Method; import java.util.List; import com.fasterxml.jackson.databind.ObjectMapper; import org.junit.jupiter.api.Test; import org.springframework.boot.autoconfigure.AutoConfigurations; -import org.springframework.boot.autoconfigure.codec.CodecProperties; +import org.springframework.boot.test.context.assertj.AssertableWebApplicationContext; import org.springframework.boot.test.context.runner.WebApplicationContextRunner; import org.springframework.boot.web.codec.CodecCustomizer; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; -import org.springframework.core.annotation.AnnotationAwareOrderComparator; +import org.springframework.core.Ordered; import org.springframework.http.codec.CodecConfigurer; +import org.springframework.http.codec.CodecConfigurer.DefaultCodecs; import org.springframework.http.codec.support.DefaultClientCodecConfigurer; -import org.springframework.util.ReflectionUtils; import static org.assertj.core.api.Assertions.assertThat; @@ -48,34 +47,58 @@ class CodecsAutoConfigurationTests { @Test void autoConfigShouldProvideALoggingRequestDetailsCustomizer() { - this.contextRunner.run((context) -> { - CodecCustomizer customizer = context.getBean(CodecCustomizer.class); - CodecConfigurer configurer = new DefaultClientCodecConfigurer(); - customizer.customize(configurer); - assertThat(configurer.defaultCodecs()).hasFieldOrPropertyWithValue("enableLoggingRequestDetails", false); - }); + this.contextRunner.run((context) -> assertThat(defaultCodecs(context)) + .hasFieldOrPropertyWithValue("enableLoggingRequestDetails", false)); + } + + @Test + void loggingRequestDetailsCustomizerShouldUseCodecProperties() { + this.contextRunner.withPropertyValues("spring.codec.log-request-details=true") + .run((context) -> assertThat(defaultCodecs(context)) + .hasFieldOrPropertyWithValue("enableLoggingRequestDetails", true)); + } + + @Test + void loggingRequestDetailsCustomizerShouldUseHttpCodecsProperties() { + this.contextRunner.withPropertyValues("spring.http.codecs.log-request-details=true") + .run((context) -> assertThat(defaultCodecs(context)) + .hasFieldOrPropertyWithValue("enableLoggingRequestDetails", true)); + } + + @Test + void logRequestDetailsShouldGivePriorityToHttpCodecProperty() { + this.contextRunner + .withPropertyValues("spring.http.codecs.log-request-details=true", "spring.codec.log-request-details=false") + .run((context) -> assertThat(defaultCodecs(context)) + .hasFieldOrPropertyWithValue("enableLoggingRequestDetails", true)); + } + @Test + void maxInMemorySizeShouldUseCodecProperties() { + this.contextRunner.withPropertyValues("spring.codec.max-in-memory-size=64KB") + .run((context) -> assertThat(defaultCodecs(context)).hasFieldOrPropertyWithValue("maxInMemorySize", + 64 * 1024)); + } + + @Test + void maxInMemorySizeShouldUseHttpCodecProperties() { + this.contextRunner.withPropertyValues("spring.http.codecs.max-in-memory-size=64KB") + .run((context) -> assertThat(defaultCodecs(context)).hasFieldOrPropertyWithValue("maxInMemorySize", + 64 * 1024)); } @Test - void loggingRequestDetailsCustomizerShouldUseHttpProperties() { - this.contextRunner.withPropertyValues("spring.codec.log-request-details=true").run((context) -> { - CodecCustomizer customizer = context.getBean(CodecCustomizer.class); - CodecConfigurer configurer = new DefaultClientCodecConfigurer(); - customizer.customize(configurer); - assertThat(configurer.defaultCodecs()).hasFieldOrPropertyWithValue("enableLoggingRequestDetails", true); - }); + void maxInMemorySizeShouldGivePriorityToHttpCodecProperty() { + this.contextRunner + .withPropertyValues("spring.http.codecs.max-in-memory-size=64KB", "spring.codec.max-in-memory-size=32KB") + .run((context) -> assertThat(defaultCodecs(context)).hasFieldOrPropertyWithValue("maxInMemorySize", + 64 * 1024)); } @Test void defaultCodecCustomizerBeanShouldHaveOrderZero() { - this.contextRunner.run((context) -> { - Method customizerMethod = ReflectionUtils.findMethod( - CodecsAutoConfiguration.DefaultCodecsConfiguration.class, "defaultCodecCustomizer", - CodecProperties.class); - Integer order = new TestAnnotationAwareOrderComparator().findOrder(customizerMethod); - assertThat(order).isZero(); - }); + this.contextRunner + .run((context) -> assertThat(context.getBean("defaultCodecCustomizer", Ordered.class).getOrder()).isZero()); } @Test @@ -101,21 +124,16 @@ void userProvidedCustomizerCanOverrideJacksonCodecCustomizer() { @Test void maxInMemorySizeEnforcedInDefaultCodecs() { - this.contextRunner.withPropertyValues("spring.codec.max-in-memory-size=1MB").run((context) -> { - CodecCustomizer customizer = context.getBean(CodecCustomizer.class); - CodecConfigurer configurer = new DefaultClientCodecConfigurer(); - customizer.customize(configurer); - assertThat(configurer.defaultCodecs()).hasFieldOrPropertyWithValue("maxInMemorySize", 1048576); - }); + this.contextRunner.withPropertyValues("spring.codec.max-in-memory-size=1MB") + .run((context) -> assertThat(defaultCodecs(context)).hasFieldOrPropertyWithValue("maxInMemorySize", + 1048576)); } - static class TestAnnotationAwareOrderComparator extends AnnotationAwareOrderComparator { - - @Override - public Integer findOrder(Object obj) { - return super.findOrder(obj); - } - + private DefaultCodecs defaultCodecs(AssertableWebApplicationContext context) { + CodecCustomizer customizer = context.getBean(CodecCustomizer.class); + CodecConfigurer configurer = new DefaultClientCodecConfigurer(); + customizer.customize(configurer); + return configurer.defaultCodecs(); } @Configuration(proxyBeanMethods = false) diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/integration/IntegrationAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/integration/IntegrationAutoConfigurationTests.java index 8ab1525fd768..95ea34deac33 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/integration/IntegrationAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/integration/IntegrationAutoConfigurationTests.java @@ -62,6 +62,8 @@ import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Primary; import org.springframework.core.task.SimpleAsyncTaskExecutor; +import org.springframework.core.task.SyncTaskExecutor; +import org.springframework.core.task.TaskExecutor; import org.springframework.integration.annotation.IntegrationComponentScan; import org.springframework.integration.annotation.MessagingGateway; import org.springframework.integration.annotation.ServiceActivator; @@ -105,6 +107,7 @@ * @author Stephane Nicoll * @author Vedran Pavic * @author Yong-Hyun Kim + * @author Yanming Zhou */ class IntegrationAutoConfigurationTests { @@ -543,6 +546,19 @@ void integrationVirtualThreadsEnabled() { .usesVirtualThreads())); } + @Test + void pollerMetadataCanBeCustomizedViaPollerMetadataCustomizer() { + TaskExecutor taskExecutor = new SyncTaskExecutor(); + this.contextRunner.withUserConfiguration(PollingConsumerConfiguration.class) + .withBean(PollerMetadataCustomizer.class, + () -> (pollerMetadata) -> pollerMetadata.setTaskExecutor(taskExecutor)) + .run((context) -> { + assertThat(context).hasSingleBean(PollerMetadata.class); + PollerMetadata metadata = context.getBean(PollerMetadata.DEFAULT_POLLER, PollerMetadata.class); + assertThat(metadata.getTaskExecutor()).isSameAs(taskExecutor); + }); + } + @Configuration(proxyBeanMethods = false) static class CustomMBeanExporter { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/integration/IntegrationPropertiesEnvironmentPostProcessorTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/integration/IntegrationPropertiesEnvironmentPostProcessorTests.java index 4cacb3e54df4..923663232dd5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/integration/IntegrationPropertiesEnvironmentPostProcessorTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/integration/IntegrationPropertiesEnvironmentPostProcessorTests.java @@ -25,7 +25,6 @@ import java.util.Map; import java.util.function.Consumer; -import io.lettuce.core.dynamic.support.ReflectionUtils; import org.junit.jupiter.api.Test; import org.junit.jupiter.params.ParameterizedTest; import org.junit.jupiter.params.provider.MethodSource; @@ -48,6 +47,7 @@ import org.springframework.mock.env.MockEnvironment; import org.springframework.test.util.ReflectionTestUtils; import org.springframework.util.ClassUtils; +import org.springframework.util.ReflectionUtils; import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatIllegalStateException; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jackson/JacksonAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jackson/JacksonAutoConfigurationTests.java index 29dde38a9714..52f5965a81c4 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jackson/JacksonAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jackson/JacksonAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -73,6 +73,7 @@ import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Import; import org.springframework.context.annotation.Primary; +import org.springframework.core.annotation.Order; import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder; import static org.assertj.core.api.Assertions.assertThat; @@ -324,6 +325,17 @@ void moduleBeansAndWellKnownModulesAreRegisteredWithTheObjectMapperBuilder() { }); } + @Test + void customModulesRegisteredByBuilderCustomizerShouldBeRetained() { + this.contextRunner.withUserConfiguration(ModuleConfig.class, CustomModuleBuilderCustomizerConfig.class) + .run((context) -> { + ObjectMapper objectMapper = context.getBean(Jackson2ObjectMapperBuilder.class).build(); + assertThat(context.getBean(CustomModule.class).getOwners()).contains(objectMapper); + assertThat(objectMapper.getRegisteredModuleIds()).contains("module-A", "module-B", + CustomModule.class.getName()); + }); + } + @Test void defaultSerializationInclusion() { this.contextRunner.run((context) -> { @@ -592,6 +604,23 @@ Jackson2ObjectMapperBuilderCustomizer customDateFormat() { } + @Configuration(proxyBeanMethods = false) + static class CustomModuleBuilderCustomizerConfig { + + @Bean + @Order(-1) + Jackson2ObjectMapperBuilderCustomizer highPrecedenceCustomizer() { + return (builder) -> builder.modulesToInstall((modules) -> modules.add(new SimpleModule("module-A"))); + } + + @Bean + @Order(1) + Jackson2ObjectMapperBuilderCustomizer lowPrecedenceCustomizer() { + return (builder) -> builder.modulesToInstall((modules) -> modules.add(new SimpleModule("module-B"))); + } + + } + @Configuration(proxyBeanMethods = false) static class ObjectMapperBuilderConsumerConfig { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/DataSourceAutoConfigurationWithoutSpringJdbcTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/DataSourceAutoConfigurationWithoutSpringJdbcTests.java new file mode 100644 index 000000000000..9f0c5ece982e --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/DataSourceAutoConfigurationWithoutSpringJdbcTests.java @@ -0,0 +1,72 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.jdbc; + +import java.util.Random; +import java.util.function.Function; + +import javax.sql.DataSource; + +import com.zaxxer.hikari.HikariDataSource; +import org.junit.jupiter.api.Test; + +import org.springframework.boot.autoconfigure.AutoConfigurations; +import org.springframework.boot.test.context.FilteredClassLoader; +import org.springframework.boot.test.context.runner.ApplicationContextRunner; +import org.springframework.boot.testsupport.classpath.ClassPathExclusions; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link DataSourceAutoConfiguration} without spring-jdbc on the classpath. + * + * @author Andy Wilkinson + */ +@ClassPathExclusions("spring-jdbc-*.jar") +class DataSourceAutoConfigurationWithoutSpringJdbcTests { + + private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() + .withConfiguration(AutoConfigurations.of(DataSourceAutoConfiguration.class)); + + @Test + void pooledDataSourceCanBeAutoConfigured() { + this.contextRunner.run((context) -> { + HikariDataSource dataSource = context.getBean(HikariDataSource.class); + assertThat(dataSource.getJdbcUrl()).isNotNull(); + assertThat(dataSource.getDriverClassName()).isNotNull(); + }); + } + + @Test + void withoutConnectionPoolsAutoConfigurationBacksOff() { + this.contextRunner.with(hideConnectionPools()) + .run((context) -> assertThat(context).doesNotHaveBean(DataSource.class)); + } + + @Test + void withUrlAndWithoutConnectionPoolsAutoConfigurationBacksOff() { + this.contextRunner.with(hideConnectionPools()) + .withPropertyValues("spring.datasource.url:jdbc:hsqldb:mem:testdb-" + new Random().nextInt()) + .run((context) -> assertThat(context).doesNotHaveBean(DataSource.class)); + } + + private static Function hideConnectionPools() { + return (runner) -> runner.withClassLoader(new FilteredClassLoader("org.apache.tomcat", "com.zaxxer.hikari", + "org.apache.commons.dbcp2", "oracle.ucp.jdbc", "com.mchange")); + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/DataSourceJmxConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/DataSourceJmxConfigurationTests.java index b8c365a8f54f..5b97ffc02317 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/DataSourceJmxConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/DataSourceJmxConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -149,7 +149,7 @@ void tomcatDoesNotExposeMBeanPoolByDefault() { @Test void tomcatAutoConfiguredCanExposeMBeanPool() { this.contextRunner - .withPropertyValues("spring.datasource.type=" + DataSource.class.getName(), + .withPropertyValues("spring.jmx.enabled=true", "spring.datasource.type=" + DataSource.class.getName(), "spring.datasource.tomcat.jmx-enabled=true") .run((context) -> { assertThat(context).hasBean("dataSourceMBean"); @@ -162,7 +162,7 @@ void tomcatAutoConfiguredCanExposeMBeanPool() { @Test void tomcatProxiedCanExposeMBeanPool() { this.contextRunner.withUserConfiguration(DataSourceProxyConfiguration.class) - .withPropertyValues("spring.datasource.type=" + DataSource.class.getName(), + .withPropertyValues("spring.jmx.enabled=true", "spring.datasource.type=" + DataSource.class.getName(), "spring.datasource.tomcat.jmx-enabled=true") .run((context) -> { assertThat(context).hasBean("dataSourceMBean"); @@ -173,7 +173,7 @@ void tomcatProxiedCanExposeMBeanPool() { @Test void tomcatDelegateCanExposeMBeanPool() { this.contextRunner.withUserConfiguration(DataSourceDelegateConfiguration.class) - .withPropertyValues("spring.datasource.type=" + DataSource.class.getName(), + .withPropertyValues("spring.jmx.enabled=true", "spring.datasource.type=" + DataSource.class.getName(), "spring.datasource.tomcat.jmx-enabled=true") .run((context) -> { assertThat(context).hasBean("dataSourceMBean"); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/HikariDataSourceConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/HikariDataSourceConfigurationTests.java index 3019be543757..4841e2a557e6 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/HikariDataSourceConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/HikariDataSourceConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,6 +16,12 @@ package org.springframework.boot.autoconfigure.jdbc; +import java.io.PrintWriter; +import java.sql.Connection; +import java.sql.SQLException; +import java.sql.SQLFeatureNotSupportedException; +import java.util.logging.Logger; + import javax.sql.DataSource; import com.zaxxer.hikari.HikariDataSource; @@ -28,12 +34,15 @@ import org.springframework.boot.jdbc.DataSourceBuilder; import org.springframework.boot.jdbc.HikariCheckpointRestoreLifecycle; import org.springframework.boot.test.context.runner.ApplicationContextRunner; +import org.springframework.boot.testsupport.classpath.ClassPathExclusions; import org.springframework.boot.testsupport.classpath.ClassPathOverrides; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.jdbc.datasource.DelegatingDataSource; import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.assertThatNoException; +import static org.mockito.Mockito.mock; /** * Tests for {@link DataSourceAutoConfiguration} with Hikari. @@ -80,7 +89,33 @@ void testDataSourceGenericPropertiesOverridden() { HikariDataSource ds = context.getBean(HikariDataSource.class); assertThat(ds.getDataSourceProperties().getProperty("dataSourceClassName")) .isEqualTo("org.h2.JDBCDataSource"); + }); + } + + @Test + @SuppressWarnings("resource") + @ClassPathExclusions({ "h2-*.jar", "hsqldb-*.jar" }) + void configureDataSourceClassNameWithNoEmbeddedDatabaseAvailable() { + this.contextRunner + .withPropertyValues("spring.datasource.url=jdbc:example//", + "spring.datasource.hikari.data-source-class-name=" + MockDataSource.class.getName()) + .run((context) -> { + HikariDataSource ds = context.getBean(HikariDataSource.class); + assertThat(ds.getDataSourceClassName()).isEqualTo(MockDataSource.class.getName()); + assertThatNoException().isThrownBy(() -> ds.getConnection().close()); + }); + } + @Test + @SuppressWarnings("resource") + void configureDataSourceClassNameToOverrideUseOfAnEmbeddedDatabase() { + this.contextRunner + .withPropertyValues("spring.datasource.url=jdbc:example//", + "spring.datasource.hikari.data-source-class-name=" + MockDataSource.class.getName()) + .run((context) -> { + HikariDataSource ds = context.getBean(HikariDataSource.class); + assertThat(ds.getDataSourceClassName()).isEqualTo(MockDataSource.class.getName()); + assertThatNoException().isThrownBy(() -> ds.getConnection().close()); }); } @@ -201,4 +236,51 @@ DataSource dataSource() { } + public static class MockDataSource implements DataSource { + + @Override + public Logger getParentLogger() throws SQLFeatureNotSupportedException { + return null; + } + + @Override + public T unwrap(Class iface) throws SQLException { + return null; + } + + @Override + public boolean isWrapperFor(Class iface) throws SQLException { + return false; + } + + @Override + public Connection getConnection() throws SQLException { + return mock(Connection.class); + } + + @Override + public Connection getConnection(String username, String password) throws SQLException { + return getConnection(); + } + + @Override + public PrintWriter getLogWriter() throws SQLException { + return null; + } + + @Override + public void setLogWriter(PrintWriter out) throws SQLException { + } + + @Override + public void setLoginTimeout(int seconds) throws SQLException { + } + + @Override + public int getLoginTimeout() throws SQLException { + return -1; + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/HikariJdbcConnectionDetailsBeanPostProcessorTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/HikariJdbcConnectionDetailsBeanPostProcessorTests.java index 3172a1b5bbf4..5bd84e10cad8 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/HikariJdbcConnectionDetailsBeanPostProcessorTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/HikariJdbcConnectionDetailsBeanPostProcessorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,6 +22,7 @@ import org.springframework.boot.jdbc.DatabaseDriver; import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.Mockito.mock; /** * Tests for {@link HikariJdbcConnectionDetailsBeanPostProcessor}. @@ -47,4 +48,13 @@ void setUsernamePasswordAndUrl() { assertThat(dataSource.getDriverClassName()).isEqualTo(DatabaseDriver.POSTGRESQL.getDriverClassName()); } + @Test + void toleratesConnectionDetailsWithNullDriverClassName() { + HikariDataSource dataSource = new HikariDataSource(); + dataSource.setDriverClassName(DatabaseDriver.H2.getDriverClassName()); + JdbcConnectionDetails connectionDetails = mock(JdbcConnectionDetails.class); + new HikariJdbcConnectionDetailsBeanPostProcessor(null).processDataSource(dataSource, connectionDetails); + assertThat(dataSource.getDriverClassName()).isEqualTo(DatabaseDriver.H2.getDriverClassName()); + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/JdbcTemplateAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/JdbcTemplateAutoConfigurationTests.java index 9326b731310a..30fd043d07fa 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/JdbcTemplateAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/JdbcTemplateAutoConfigurationTests.java @@ -39,6 +39,8 @@ import org.springframework.jdbc.core.JdbcTemplate; import org.springframework.jdbc.core.namedparam.NamedParameterJdbcOperations; import org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate; +import org.springframework.jdbc.support.SQLExceptionTranslator; +import org.springframework.jdbc.support.SQLStateSQLExceptionTranslator; import static org.assertj.core.api.Assertions.assertThat; import static org.mockito.Mockito.mock; @@ -64,24 +66,35 @@ void testJdbcTemplateExists() { assertThat(context).hasSingleBean(JdbcOperations.class); JdbcTemplate jdbcTemplate = context.getBean(JdbcTemplate.class); assertThat(jdbcTemplate.getDataSource()).isEqualTo(context.getBean(DataSource.class)); + assertThat(jdbcTemplate.isIgnoreWarnings()).isEqualTo(true); assertThat(jdbcTemplate.getFetchSize()).isEqualTo(-1); assertThat(jdbcTemplate.getQueryTimeout()).isEqualTo(-1); assertThat(jdbcTemplate.getMaxRows()).isEqualTo(-1); + assertThat(jdbcTemplate.isSkipResultsProcessing()).isEqualTo(false); + assertThat(jdbcTemplate.isSkipUndeclaredResults()).isEqualTo(false); + assertThat(jdbcTemplate.isResultsMapCaseInsensitive()).isEqualTo(false); }); } @Test void testJdbcTemplateWithCustomProperties() { this.contextRunner - .withPropertyValues("spring.jdbc.template.fetch-size:100", "spring.jdbc.template.query-timeout:60", - "spring.jdbc.template.max-rows:1000") + .withPropertyValues("spring.jdbc.template.ignore-warnings:false", "spring.jdbc.template.fetch-size:100", + "spring.jdbc.template.query-timeout:60", "spring.jdbc.template.max-rows:1000", + "spring.jdbc.template.skip-results-processing:true", + "spring.jdbc.template.skip-undeclared-results:true", + "spring.jdbc.template.results-map-case-insensitive:true") .run((context) -> { assertThat(context).hasSingleBean(JdbcOperations.class); JdbcTemplate jdbcTemplate = context.getBean(JdbcTemplate.class); assertThat(jdbcTemplate.getDataSource()).isNotNull(); + assertThat(jdbcTemplate.isIgnoreWarnings()).isEqualTo(false); assertThat(jdbcTemplate.getFetchSize()).isEqualTo(100); assertThat(jdbcTemplate.getQueryTimeout()).isEqualTo(60); assertThat(jdbcTemplate.getMaxRows()).isEqualTo(1000); + assertThat(jdbcTemplate.isSkipResultsProcessing()).isEqualTo(true); + assertThat(jdbcTemplate.isSkipUndeclaredResults()).isEqualTo(true); + assertThat(jdbcTemplate.isResultsMapCaseInsensitive()).isEqualTo(true); }); } @@ -238,6 +251,31 @@ void testDependencyToLiquibaseWithJdbcTemplateMixed() { }); } + @Test + void shouldConfigureJdbcTemplateWithSQLExceptionTranslatorIfPresent() { + SQLStateSQLExceptionTranslator sqlExceptionTranslator = new SQLStateSQLExceptionTranslator(); + this.contextRunner.withBean(SQLExceptionTranslator.class, () -> sqlExceptionTranslator).run((context) -> { + assertThat(context).hasSingleBean(JdbcTemplate.class); + JdbcTemplate jdbcTemplate = context.getBean(JdbcTemplate.class); + assertThat(jdbcTemplate.getExceptionTranslator()).isSameAs(sqlExceptionTranslator); + }); + } + + @Test + void shouldNotConfigureJdbcTemplateWithSQLExceptionTranslatorIfNotUnique() { + SQLStateSQLExceptionTranslator sqlExceptionTranslator1 = new SQLStateSQLExceptionTranslator(); + SQLStateSQLExceptionTranslator sqlExceptionTranslator2 = new SQLStateSQLExceptionTranslator(); + this.contextRunner + .withBean("sqlExceptionTranslator1", SQLExceptionTranslator.class, () -> sqlExceptionTranslator1) + .withBean("sqlExceptionTranslator2", SQLExceptionTranslator.class, () -> sqlExceptionTranslator2) + .run((context) -> { + assertThat(context).hasSingleBean(JdbcTemplate.class); + JdbcTemplate jdbcTemplate = context.getBean(JdbcTemplate.class); + assertThat(jdbcTemplate.getExceptionTranslator()).isNotSameAs(sqlExceptionTranslator1) + .isNotSameAs(sqlExceptionTranslator2); + }); + } + @Configuration(proxyBeanMethods = false) static class CustomConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/TomcatDataSourceConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/TomcatDataSourceConfigurationTests.java index fc4ef5e00238..ba5c8abd8bc8 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/TomcatDataSourceConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jdbc/TomcatDataSourceConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -141,7 +141,7 @@ void usesCustomJdbcConnectionDetailsWhenDefined() { static class TomcatDataSourceConfiguration { @Bean - @ConfigurationProperties(prefix = "spring.datasource.tomcat") + @ConfigurationProperties("spring.datasource.tomcat") DataSource dataSource() { return DataSourceBuilder.create().type(org.apache.tomcat.jdbc.pool.DataSource.class).build(); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jms/AcknowledgeModeTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jms/AcknowledgeModeTests.java index cf785f750657..bdfba2ef2422 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jms/AcknowledgeModeTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jms/AcknowledgeModeTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/DefaultExceptionTranslatorExecuteListenerTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/DefaultExceptionTranslatorExecuteListenerTests.java index ae1278889d1b..c4c36b20ff86 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/DefaultExceptionTranslatorExecuteListenerTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/DefaultExceptionTranslatorExecuteListenerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,7 +53,7 @@ void createWhenTranslatorFactoryIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new DefaultExceptionTranslatorExecuteListener( (Function) null)) - .withMessage("TranslatorFactory must not be null"); + .withMessage("'translatorFactory' must not be null"); } @ParameterizedTest(name = "{0}") diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/JooqAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/JooqAutoConfigurationTests.java index 8e8025f3e6f5..0bca37db8edd 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/JooqAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/JooqAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,6 +28,7 @@ import org.jooq.TransactionContext; import org.jooq.TransactionProvider; import org.jooq.TransactionalRunnable; +import org.jooq.conf.Settings; import org.jooq.impl.DataSourceConnectionProvider; import org.jooq.impl.DefaultDSLContext; import org.jooq.impl.DefaultExecuteListenerProvider; @@ -35,6 +36,7 @@ import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.jdbc.DataSourceBuilder; +import org.springframework.boot.test.context.FilteredClassLoader; import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -57,6 +59,7 @@ * @author Stephane Nicoll * @author Dmytro Nosan * @author Dennis Melzer + * @author Moritz Halbritter */ class JooqAutoConfigurationTests { @@ -221,6 +224,37 @@ void autoConfiguredJooqConfigurationCanBeUsedToCreateCustomDslContext() { .run((context) -> assertThat(context).hasSingleBean(DSLContext.class).hasBean("customDslContext")); } + @Test + void shouldLoadSettingsFromConfigPropertyThroughJaxb() { + this.contextRunner.withUserConfiguration(JooqDataSourceConfiguration.class) + .withPropertyValues("spring.jooq.config=classpath:org/springframework/boot/autoconfigure/jooq/settings.xml") + .run((context) -> { + assertThat(context).hasSingleBean(Settings.class); + Settings settings = context.getBean(Settings.class); + assertThat(settings.getBatchSize()).isEqualTo(100); + }); + } + + @Test + void shouldNotProvideSettingsIfJaxbIsMissing() { + this.contextRunner.withUserConfiguration(JooqDataSourceConfiguration.class) + .withClassLoader(new FilteredClassLoader("jakarta.xml.bind")) + .withPropertyValues("spring.jooq.config=classpath:org/springframework/boot/autoconfigure/jooq/settings.xml") + .run((context) -> assertThat(context).hasFailed() + .getFailure() + .hasRootCauseInstanceOf(JaxbNotAvailableException.class)); + } + + @Test + void shouldFailWithSensibleErrorMessageIfConfigIsNotFound() { + this.contextRunner.withUserConfiguration(JooqDataSourceConfiguration.class) + .withPropertyValues("spring.jooq.config=classpath:does-not-exist.xml") + .run((context) -> assertThat(context).hasFailed() + .getFailure() + .hasMessageContaining("spring.jooq.config") + .hasMessageContaining("does-not-exist.xml")); + } + static class AssertFetch implements TransactionalRunnable { private final DSLContext dsl; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/JooqExceptionTranslatorTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/JooqExceptionTranslatorTests.java deleted file mode 100644 index f5390c0f5e61..000000000000 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/jooq/JooqExceptionTranslatorTests.java +++ /dev/null @@ -1,91 +0,0 @@ -/* - * Copyright 2012-2024 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.autoconfigure.jooq; - -import java.sql.SQLException; - -import org.jooq.Configuration; -import org.jooq.ExecuteContext; -import org.jooq.SQLDialect; -import org.junit.jupiter.api.Test; -import org.junit.jupiter.params.ParameterizedTest; -import org.junit.jupiter.params.provider.MethodSource; - -import org.springframework.jdbc.BadSqlGrammarException; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.mockito.ArgumentMatchers.any; -import static org.mockito.ArgumentMatchers.assertArg; -import static org.mockito.BDDMockito.given; -import static org.mockito.BDDMockito.then; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.never; - -/** - * Tests for {@link JooqExceptionTranslator} - * - * @author Andy Wilkinson - * @deprecated since 3.3.0 for removal in 3.5.0 - */ -@Deprecated(since = "3.3.0") -@SuppressWarnings("removal") -class JooqExceptionTranslatorTests { - - private final JooqExceptionTranslator exceptionTranslator = new JooqExceptionTranslator(); - - @ParameterizedTest(name = "{0}") - @MethodSource - void exceptionTranslation(SQLDialect dialect, SQLException sqlException) { - ExecuteContext context = mock(ExecuteContext.class); - Configuration configuration = mock(Configuration.class); - given(context.configuration()).willReturn(configuration); - given(configuration.dialect()).willReturn(dialect); - given(context.sqlException()).willReturn(sqlException); - this.exceptionTranslator.exception(context); - then(context).should().exception(assertArg((ex) -> assertThat(ex).isInstanceOf(BadSqlGrammarException.class))); - } - - @Test - void whenExceptionCannotBeTranslatedThenExecuteContextExceptionIsNotCalled() { - ExecuteContext context = mock(ExecuteContext.class); - Configuration configuration = mock(Configuration.class); - given(context.configuration()).willReturn(configuration); - given(configuration.dialect()).willReturn(SQLDialect.POSTGRES); - given(context.sqlException()).willReturn(new SQLException(null, null, 123456789)); - this.exceptionTranslator.exception(context); - then(context).should(never()).exception(any()); - } - - static Object[] exceptionTranslation() { - return new Object[] { new Object[] { SQLDialect.DERBY, sqlException("42802") }, - new Object[] { SQLDialect.H2, sqlException(42000) }, - new Object[] { SQLDialect.HSQLDB, sqlException(-22) }, - new Object[] { SQLDialect.MARIADB, sqlException(1054) }, - new Object[] { SQLDialect.MYSQL, sqlException(1054) }, - new Object[] { SQLDialect.POSTGRES, sqlException("03000") }, - new Object[] { SQLDialect.SQLITE, sqlException("21000") } }; - } - - private static SQLException sqlException(String sqlState) { - return new SQLException(null, sqlState); - } - - private static SQLException sqlException(int vendorCode) { - return new SQLException(null, null, vendorCode); - } - -} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/ConcurrentKafkaListenerContainerFactoryConfigurerTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/ConcurrentKafkaListenerContainerFactoryConfigurerTests.java index 779d9974d3f1..42d9c1a90370 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/ConcurrentKafkaListenerContainerFactoryConfigurerTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/ConcurrentKafkaListenerContainerFactoryConfigurerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,6 +16,7 @@ package org.springframework.boot.autoconfigure.kafka; +import java.time.Duration; import java.util.function.Function; import org.junit.jupiter.api.BeforeEach; @@ -80,4 +81,12 @@ void shouldApplyListenerTaskExecutor() { assertThat(this.factory.getContainerProperties().getListenerTaskExecutor()).isEqualTo(executor); } + @Test + void shouldApplyAuthExceptionRetryInterval() { + this.properties.getListener().setAuthExceptionRetryInterval(Duration.ofSeconds(10)); + this.configurer.configure(this.factory, this.consumerFactory); + assertThat(this.factory.getContainerProperties().getAuthExceptionRetryInterval()) + .isEqualTo(Duration.ofSeconds(10)); + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/KafkaAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/KafkaAutoConfigurationTests.java index 05afc38597b8..49e32c02369c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/KafkaAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/KafkaAutoConfigurationTests.java @@ -50,6 +50,8 @@ import org.springframework.aot.hint.predicate.RuntimeHintsPredicates; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.ssl.SslAutoConfiguration; +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslStoreBundle; import org.springframework.boot.test.context.assertj.AssertableApplicationContext; import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.boot.test.context.runner.ContextConsumer; @@ -113,6 +115,7 @@ * @author Andy Wilkinson * @author Phillip Webb * @author Scott Frederick + * @author Yanming Zhou */ class KafkaAutoConfigurationTests { @@ -130,8 +133,9 @@ void consumerProperties() { "spring.kafka.ssl.key-store-type=PKCS12", "spring.kafka.ssl.trust-store-location=classpath:tsLoc", "spring.kafka.ssl.trust-store-password=p3", "spring.kafka.ssl.trust-store-type=PKCS12", "spring.kafka.ssl.protocol=TLSv1.2", "spring.kafka.consumer.auto-commit-interval=123", - "spring.kafka.consumer.max-poll-records=42", "spring.kafka.consumer.auto-offset-reset=earliest", - "spring.kafka.consumer.client-id=ccid", // test override common + "spring.kafka.consumer.max-poll-records=42", "spring.kafka.consumer.max-poll-interval=30s", + "spring.kafka.consumer.auto-offset-reset=earliest", "spring.kafka.consumer.client-id=ccid", + // test override common "spring.kafka.consumer.enable-auto-commit=false", "spring.kafka.consumer.fetch-max-wait=456", "spring.kafka.consumer.properties.fiz.buz=fix.fox", "spring.kafka.consumer.fetch-min-size=1KB", "spring.kafka.consumer.group-id=bar", "spring.kafka.consumer.heartbeat-interval=234", @@ -170,6 +174,7 @@ void consumerProperties() { assertThat(configs).containsEntry(ConsumerConfig.VALUE_DESERIALIZER_CLASS_CONFIG, IntegerDeserializer.class); assertThat(configs).containsEntry(ConsumerConfig.MAX_POLL_RECORDS_CONFIG, 42); + assertThat(configs).containsEntry(ConsumerConfig.MAX_POLL_INTERVAL_MS_CONFIG, 30000); assertThat(configs).containsEntry("foo", "bar"); assertThat(configs).containsEntry("baz", "qux"); assertThat(configs).containsEntry("foo.bar.baz", "qux.fiz.buz"); @@ -198,10 +203,33 @@ void connectionDetailsAreAppliedToConsumer() { Collections.singletonList("kafka.example.com:12345")); assertThat(configs).containsEntry(ConsumerConfig.BOOTSTRAP_SERVERS_CONFIG, Collections.singletonList("kafka.example.com:12345")); - assertThat(configs).containsEntry(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); }); } + @Test + void connectionDetailsWithSslBundleAreAppliedToConsumer() { + SslBundle sslBundle = SslBundle.of(SslStoreBundle.NONE); + KafkaConnectionDetails connectionDetails = new KafkaConnectionDetails() { + @Override + public List getBootstrapServers() { + return List.of("kafka.example.com:12345"); + } + + @Override + public Configuration getConsumer() { + return Configuration.of(getBootstrapServers(), sslBundle); + } + + }; + this.contextRunner.withBean(KafkaConnectionDetails.class, () -> connectionDetails).run((context) -> { + assertThat(context).hasSingleBean(KafkaConnectionDetails.class); + DefaultKafkaConsumerFactory consumerFactory = context.getBean(DefaultKafkaConsumerFactory.class); + Map configs = consumerFactory.getConfigurationProperties(); + assertThat(configs).containsEntry("ssl.engine.factory.class", SslBundleSslEngineFactory.class); + assertThat(configs).containsEntry("org.springframework.boot.ssl.SslBundle", sslBundle); + }); + } + @Test @WithResource(name = "ksLocP") @WithResource(name = "tsLocP") @@ -270,10 +298,33 @@ void connectionDetailsAreAppliedToProducer() { Collections.singletonList("kafka.example.com:12345")); assertThat(configs).containsEntry(ProducerConfig.BOOTSTRAP_SERVERS_CONFIG, Collections.singletonList("kafka.example.com:12345")); - assertThat(configs).containsEntry(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); }); } + @Test + void connectionDetailsWithSslBundleAreAppliedToProducer() { + SslBundle sslBundle = SslBundle.of(SslStoreBundle.NONE); + KafkaConnectionDetails connectionDetails = new KafkaConnectionDetails() { + @Override + public List getBootstrapServers() { + return List.of("kafka.example.com:12345"); + } + + @Override + public Configuration getProducer() { + return Configuration.of(getBootstrapServers(), sslBundle); + } + + }; + this.contextRunner.withBean(KafkaConnectionDetails.class, () -> connectionDetails).run((context) -> { + assertThat(context).hasSingleBean(KafkaConnectionDetails.class); + DefaultKafkaProducerFactory producerFactory = context.getBean(DefaultKafkaProducerFactory.class); + Map configs = producerFactory.getConfigurationProperties(); + assertThat(configs).containsEntry("ssl.engine.factory.class", SslBundleSslEngineFactory.class); + assertThat(configs).containsEntry("org.springframework.boot.ssl.SslBundle", sslBundle); + }); + } + @Test @WithResource(name = "ksLocP") @WithResource(name = "tsLocP") @@ -332,11 +383,33 @@ void connectionDetailsAreAppliedToAdmin() { Collections.singletonList("kafka.example.com:12345")); assertThat(configs).containsEntry(AdminClientConfig.BOOTSTRAP_SERVERS_CONFIG, Collections.singletonList("kafka.example.com:12345")); - assertThat(configs).containsEntry(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); - assertThat(configs).containsEntry(AdminClientConfig.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); }); } + @Test + void connectionDetailsWithSslBundleAreAppliedToAdmin() { + SslBundle sslBundle = SslBundle.of(SslStoreBundle.NONE); + KafkaConnectionDetails connectionDetails = new KafkaConnectionDetails() { + @Override + public List getBootstrapServers() { + return List.of("kafka.example.com:12345"); + } + + @Override + public Configuration getAdmin() { + return Configuration.of(getBootstrapServers(), sslBundle); + } + + }; + this.contextRunner.withBean(KafkaConnectionDetails.class, () -> connectionDetails).run((context) -> { + assertThat(context).hasSingleBean(KafkaConnectionDetails.class); + KafkaAdmin admin = context.getBean(KafkaAdmin.class); + Map configs = admin.getConfigurationProperties(); + assertThat(configs).containsEntry("ssl.engine.factory.class", SslBundleSslEngineFactory.class); + assertThat(configs).containsEntry("org.springframework.boot.ssl.SslBundle", sslBundle); + }); + } + @Test @SuppressWarnings("unchecked") @WithResource(name = "ksLocP") @@ -403,8 +476,34 @@ void connectionDetailsAreAppliedToStreams() { Collections.singletonList("kafka.example.com:12345")); assertThat(configs).containsEntry(StreamsConfig.BOOTSTRAP_SERVERS_CONFIG, Collections.singletonList("kafka.example.com:12345")); - assertThat(configs).containsEntry(CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); - assertThat(configs).containsEntry(StreamsConfig.SECURITY_PROTOCOL_CONFIG, "PLAINTEXT"); + }); + } + + @Test + void connectionDetailsWithSslBundleAreAppliedToStreams() { + SslBundle sslBundle = SslBundle.of(SslStoreBundle.NONE); + KafkaConnectionDetails connectionDetails = new KafkaConnectionDetails() { + @Override + public List getBootstrapServers() { + return List.of("kafka.example.com:12345"); + } + + @Override + public Configuration getStreams() { + return Configuration.of(getBootstrapServers(), sslBundle); + } + }; + this.contextRunner.withUserConfiguration(EnableKafkaStreamsConfiguration.class) + .withPropertyValues("spring.kafka.streams.auto-startup=false", "spring.kafka.streams.application-id=test") + .withBean(KafkaConnectionDetails.class, () -> connectionDetails) + .run((context) -> { + assertThat(context).hasSingleBean(KafkaConnectionDetails.class); + Properties configs = context + .getBean(KafkaStreamsDefaultConfiguration.DEFAULT_STREAMS_CONFIG_BEAN_NAME, + KafkaStreamsConfiguration.class) + .asProperties(); + assertThat(configs).containsEntry("ssl.engine.factory.class", SslBundleSslEngineFactory.class); + assertThat(configs).containsEntry("org.springframework.boot.ssl.SslBundle", sslBundle); }); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/KafkaPropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/KafkaPropertiesTests.java index 808bb4ba2705..a536e0786e74 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/KafkaPropertiesTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/kafka/KafkaPropertiesTests.java @@ -107,8 +107,7 @@ void sslBundleConfiguration() { properties.getSsl().setBundle("myBundle"); Map consumerProperties = properties .buildConsumerProperties(new DefaultSslBundleRegistry("myBundle", this.sslBundle)); - assertThat(consumerProperties).containsEntry(SslConfigs.SSL_ENGINE_FACTORY_CLASS_CONFIG, - SslBundleSslEngineFactory.class); + assertThat(consumerProperties).doesNotContainKey(SslConfigs.SSL_ENGINE_FACTORY_CLASS_CONFIG); } @Test @@ -117,7 +116,7 @@ void sslPropertiesWhenKeyStoreLocationAndKeySetShouldThrowException() { properties.getSsl().setKeyStoreKey("-----BEGIN"); properties.getSsl().setKeyStoreLocation(new ClassPathResource("ksLoc")); assertThatExceptionOfType(MutuallyExclusiveConfigurationPropertiesException.class) - .isThrownBy(() -> properties.buildConsumerProperties()); + .isThrownBy(properties::buildConsumerProperties); } @Test @@ -126,7 +125,7 @@ void sslPropertiesWhenTrustStoreLocationAndCertificatesSetShouldThrowException() properties.getSsl().setTrustStoreLocation(new ClassPathResource("tsLoc")); properties.getSsl().setTrustStoreCertificates("-----BEGIN"); assertThatExceptionOfType(MutuallyExclusiveConfigurationPropertiesException.class) - .isThrownBy(() -> properties.buildConsumerProperties()); + .isThrownBy(properties::buildConsumerProperties); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/ldap/LdapAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/ldap/LdapAutoConfigurationTests.java index 2d51f84a9b7b..d36350b1f6bf 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/ldap/LdapAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/ldap/LdapAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,9 +16,13 @@ package org.springframework.boot.autoconfigure.ldap; +import javax.naming.Name; + +import org.assertj.core.api.InstanceOfAssertFactories; import org.junit.jupiter.api.Test; import org.springframework.boot.autoconfigure.AutoConfigurations; +import org.springframework.boot.convert.ApplicationConversionService; import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @@ -27,6 +31,7 @@ import org.springframework.ldap.core.support.DirContextAuthenticationStrategy; import org.springframework.ldap.core.support.LdapContextSource; import org.springframework.ldap.core.support.SimpleDirContextAuthenticationStrategy; +import org.springframework.ldap.odm.core.ObjectDirectoryMapper; import org.springframework.ldap.pool2.factory.PoolConfig; import org.springframework.ldap.pool2.factory.PooledContextSource; import org.springframework.ldap.support.LdapUtils; @@ -83,6 +88,14 @@ void contextSourceWithUserDoesNotEnableAnonymousReadOnly() { }); } + @Test + void contextSourceWithReferral() { + this.contextRunner.withPropertyValues("spring.ldap.referral:ignore").run((context) -> { + LdapContextSource contextSource = context.getBean(LdapContextSource.class); + assertThat(contextSource).hasFieldOrPropertyWithValue("referral", "ignore"); + }); + } + @Test void contextSourceWithExtraCustomization() { this.contextRunner @@ -132,6 +145,20 @@ void usesCustomConnectionDetailsWhenDefined() { }); } + @Test + void objectDirectoryMapperExists() { + this.contextRunner.withPropertyValues("spring.ldap.urls:ldap://localhost:389").run((context) -> { + assertThat(context).hasSingleBean(ObjectDirectoryMapper.class); + ObjectDirectoryMapper objectDirectoryMapper = context.getBean(ObjectDirectoryMapper.class); + assertThat(objectDirectoryMapper).extracting("converterManager") + .extracting("conversionService", InstanceOfAssertFactories.type(ApplicationConversionService.class)) + .satisfies((conversionService) -> { + assertThat(conversionService.canConvert(String.class, Name.class)).isTrue(); + assertThat(conversionService.canConvert(Name.class, String.class)).isTrue(); + }); + }); + } + @Test void templateExists() { this.contextRunner.withPropertyValues("spring.ldap.urls:ldap://localhost:389").run((context) -> { @@ -140,9 +167,23 @@ void templateExists() { assertThat(ldapTemplate).hasFieldOrPropertyWithValue("ignorePartialResultException", false); assertThat(ldapTemplate).hasFieldOrPropertyWithValue("ignoreNameNotFoundException", false); assertThat(ldapTemplate).hasFieldOrPropertyWithValue("ignoreSizeLimitExceededException", true); + assertThat(ldapTemplate).extracting("objectDirectoryMapper") + .isSameAs(context.getBean(ObjectDirectoryMapper.class)); }); } + @Test + void templateCanBeConfiguredWithCustomObjectDirectoryMapper() { + ObjectDirectoryMapper objectDirectoryMapper = mock(ObjectDirectoryMapper.class); + this.contextRunner.withPropertyValues("spring.ldap.urls:ldap://localhost:389") + .withBean(ObjectDirectoryMapper.class, () -> objectDirectoryMapper) + .run((context) -> { + assertThat(context).hasSingleBean(LdapTemplate.class); + LdapTemplate ldapTemplate = context.getBean(LdapTemplate.class); + assertThat(ldapTemplate).extracting("objectDirectoryMapper").isSameAs(objectDirectoryMapper); + }); + } + @Test void templateConfigurationCanBeCustomized() { this.contextRunner diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseAutoConfigurationTests.java index 798ebf0067e9..210bb87005fe 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/liquibase/LiquibaseAutoConfigurationTests.java @@ -650,6 +650,26 @@ void whenCustomizerBeanIsDefinedThenItIsConfiguredOnSpringLiquibase() { .run(assertLiquibase((liquibase) -> assertThat(liquibase.getCustomizer()).isNotNull())); } + @Test + @WithDbChangelogMasterYamlResource + void whenAnalyticsEnabledIsFalseThenSpringLiquibaseHasAnalyticsDisabled() { + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class) + .withPropertyValues("spring.liquibase.analytics-enabled=false") + .run((context) -> assertThat(context.getBean(SpringLiquibase.class)) + .extracting(SpringLiquibase::getAnalyticsEnabled) + .isEqualTo(Boolean.FALSE)); + } + + @Test + @WithDbChangelogMasterYamlResource + void whenLicenseKeyIsSetThenSpringLiquibaseHasLicenseKey() { + this.contextRunner.withUserConfiguration(EmbeddedDataSourceConfiguration.class) + .withPropertyValues("spring.liquibase.license-key=a1b2c3d4") + .run((context) -> assertThat(context.getBean(SpringLiquibase.class)) + .extracting(SpringLiquibase::getLicenseKey) + .isEqualTo("a1b2c3d4")); + } + private ContextConsumer assertLiquibase(Consumer consumer) { return (context) -> { assertThat(context).hasSingleBean(SpringLiquibase.class); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLoggerTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLoggerTests.java index 28d8fdf3c28a..be45074dd6ae 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLoggerTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/logging/ConditionEvaluationReportLoggerTests.java @@ -58,7 +58,7 @@ void noErrorIfNotInitialized(CapturedOutput output) { void supportsOnlyInfoAndDebugLogLevels() { assertThatIllegalArgumentException() .isThrownBy(() -> new ConditionEvaluationReportLogger(LogLevel.TRACE, () -> null)) - .withMessageContaining("LogLevel must be INFO or DEBUG"); + .withMessageContaining("'logLevel' must be INFO or DEBUG"); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/MongoAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/MongoAutoConfigurationTests.java index 4ba66770174d..c853d13d83cc 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/MongoAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/MongoAutoConfigurationTests.java @@ -24,6 +24,7 @@ import com.mongodb.client.MongoClient; import com.mongodb.client.MongoClients; import com.mongodb.client.internal.MongoClientImpl; +import com.mongodb.connection.ClusterConnectionMode; import com.mongodb.connection.SslSettings; import org.junit.jupiter.api.Test; @@ -81,7 +82,7 @@ void configuresSslWhenEnabled() { this.contextRunner.withPropertyValues("spring.data.mongodb.ssl.enabled=true").run((context) -> { SslSettings sslSettings = getSettings(context).getSslSettings(); assertThat(sslSettings.isEnabled()).isTrue(); - assertThat(sslSettings.getContext()).isNull(); + assertThat(sslSettings.getContext()).isNotNull(); }); } @@ -100,6 +101,22 @@ void configuresSslWithBundle() { }); } + @Test + void configuresProtocol() { + this.contextRunner.withPropertyValues("spring.data.mongodb.protocol=mongodb+srv").run((context) -> { + MongoClientSettings settings = getSettings(context); + assertThat(settings.getClusterSettings().getMode()).isEqualTo(ClusterConnectionMode.MULTIPLE); + }); + } + + @Test + void defaultProtocol() { + this.contextRunner.run((context) -> { + MongoClientSettings settings = getSettings(context); + assertThat(settings.getClusterSettings().getMode()).isEqualTo(ClusterConnectionMode.SINGLE); + }); + } + @Test void configuresWithoutSslWhenDisabledWithBundle() { this.contextRunner diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/MongoReactiveAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/MongoReactiveAutoConfigurationTests.java index 44ee73001b15..8b4e7f067650 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/MongoReactiveAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/MongoReactiveAutoConfigurationTests.java @@ -89,7 +89,7 @@ void configuresSslWhenEnabled() { this.contextRunner.withPropertyValues("spring.data.mongodb.ssl.enabled=true").run((context) -> { SslSettings sslSettings = getSettings(context).getSslSettings(); assertThat(sslSettings.isEnabled()).isTrue(); - assertThat(sslSettings.getContext()).isNull(); + assertThat(sslSettings.getContext()).isNotNull(); }); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/PropertiesMongoConnectionDetailsTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/PropertiesMongoConnectionDetailsTests.java index 0b529d6e334c..eb9b6878b335 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/PropertiesMongoConnectionDetailsTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/mongo/PropertiesMongoConnectionDetailsTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,24 +19,41 @@ import java.util.List; import com.mongodb.ConnectionString; +import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; +import org.springframework.boot.ssl.DefaultSslBundleRegistry; +import org.springframework.boot.ssl.SslBundle; + import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.Mockito.mock; /** * Tests for {@link PropertiesMongoConnectionDetails}. * * @author Christoph Dreis * @author Scott Frederick + * @author Moritz Halbritter */ class PropertiesMongoConnectionDetailsTests { - private final MongoProperties properties = new MongoProperties(); + private MongoProperties properties; + + private DefaultSslBundleRegistry sslBundleRegistry; + + private PropertiesMongoConnectionDetails connectionDetails; + + @BeforeEach + void setUp() { + this.properties = new MongoProperties(); + this.sslBundleRegistry = new DefaultSslBundleRegistry(); + this.connectionDetails = new PropertiesMongoConnectionDetails(this.properties, this.sslBundleRegistry); + } @Test void credentialsCanBeConfiguredWithUsername() { this.properties.setUsername("user"); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getUsername()).isEqualTo("user"); assertThat(connectionString.getPassword()).isEmpty(); assertThat(connectionString.getCredential().getUserName()).isEqualTo("user"); @@ -48,7 +65,7 @@ void credentialsCanBeConfiguredWithUsername() { void credentialsCanBeConfiguredWithUsernameAndPassword() { this.properties.setUsername("user"); this.properties.setPassword("secret".toCharArray()); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getUsername()).isEqualTo("user"); assertThat(connectionString.getPassword()).isEqualTo("secret".toCharArray()); assertThat(connectionString.getCredential().getUserName()).isEqualTo("user"); @@ -59,22 +76,29 @@ void credentialsCanBeConfiguredWithUsernameAndPassword() { @Test void databaseCanBeConfigured() { this.properties.setDatabase("db"); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getDatabase()).isEqualTo("db"); } @Test void databaseHasDefaultWhenNotConfigured() { - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getDatabase()).isEqualTo("test"); } + @Test + void protocolCanBeConfigured() { + this.properties.setProtocol("mongodb+srv"); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); + assertThat(connectionString.getConnectionString()).startsWith("mongodb+srv://"); + } + @Test void authenticationDatabaseCanBeConfigured() { this.properties.setUsername("user"); this.properties.setDatabase("db"); this.properties.setAuthenticationDatabase("authdb"); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getDatabase()).isEqualTo("db"); assertThat(connectionString.getCredential().getSource()).isEqualTo("authdb"); assertThat(connectionString.getCredential().getUserName()).isEqualTo("user"); @@ -83,14 +107,14 @@ void authenticationDatabaseCanBeConfigured() { @Test void authenticationDatabaseIsNotConfiguredWhenUsernameIsNotConfigured() { this.properties.setAuthenticationDatabase("authdb"); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getCredential()).isNull(); } @Test void replicaSetCanBeConfigured() { this.properties.setReplicaSetName("test"); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getRequiredReplicaSetName()).isEqualTo("test"); } @@ -99,7 +123,7 @@ void replicaSetCanBeConfiguredWithDatabase() { this.properties.setUsername("user"); this.properties.setDatabase("db"); this.properties.setReplicaSetName("test"); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getDatabase()).isEqualTo("db"); assertThat(connectionString.getRequiredReplicaSetName()).isEqualTo("test"); } @@ -107,14 +131,14 @@ void replicaSetCanBeConfiguredWithDatabase() { @Test void replicaSetCanBeNull() { this.properties.setReplicaSetName(null); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getRequiredReplicaSetName()).isNull(); } @Test void replicaSetCanBeBlank() { this.properties.setReplicaSetName(""); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getRequiredReplicaSetName()).isNull(); } @@ -122,18 +146,32 @@ void replicaSetCanBeBlank() { void whenAdditionalHostsAreConfiguredThenTheyAreIncludedInHostsOfConnectionString() { this.properties.setHost("mongo1.example.com"); this.properties.setAdditionalHosts(List.of("mongo2.example.com", "mongo3.example.com")); - ConnectionString connectionString = getConnectionString(); + ConnectionString connectionString = this.connectionDetails.getConnectionString(); assertThat(connectionString.getHosts()).containsExactly("mongo1.example.com", "mongo2.example.com", "mongo3.example.com"); } - private PropertiesMongoConnectionDetails createConnectionDetails() { - return new PropertiesMongoConnectionDetails(this.properties); + @Test + void shouldReturnSslBundle() { + SslBundle bundle1 = mock(SslBundle.class); + this.sslBundleRegistry.registerBundle("bundle-1", bundle1); + this.properties.getSsl().setBundle("bundle-1"); + SslBundle sslBundle = this.connectionDetails.getSslBundle(); + assertThat(sslBundle).isSameAs(bundle1); + } + + @Test + void shouldReturnSystemDefaultBundleIfSslIsEnabledButBundleNotSet() { + this.properties.getSsl().setEnabled(true); + SslBundle sslBundle = this.connectionDetails.getSslBundle(); + assertThat(sslBundle).isNotNull(); } - private ConnectionString getConnectionString() { - PropertiesMongoConnectionDetails connectionDetails = createConnectionDetails(); - return connectionDetails.getConnectionString(); + @Test + void shouldReturnNullIfSslIsNotEnabled() { + this.properties.getSsl().setEnabled(false); + SslBundle sslBundle = this.connectionDetails.getSslBundle(); + assertThat(sslBundle).isNull(); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/orm/jpa/Hibernate2ndLevelCacheIntegrationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/orm/jpa/Hibernate2ndLevelCacheIntegrationTests.java index 274c373e0fc1..9696efcd87e5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/orm/jpa/Hibernate2ndLevelCacheIntegrationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/orm/jpa/Hibernate2ndLevelCacheIntegrationTests.java @@ -16,14 +16,13 @@ package org.springframework.boot.autoconfigure.orm.jpa; -import com.hazelcast.cache.impl.HazelcastServerCachingProvider; +import org.ehcache.jsr107.EhcacheCachingProvider; import org.junit.jupiter.api.Test; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.autoconfigure.cache.CacheAutoConfiguration; import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; import org.springframework.boot.test.context.runner.ApplicationContextRunner; -import org.springframework.boot.testsupport.classpath.resources.WithResource; import org.springframework.cache.annotation.EnableCaching; import org.springframework.context.annotation.Configuration; @@ -42,29 +41,12 @@ class Hibernate2ndLevelCacheIntegrationTests { .withUserConfiguration(TestConfiguration.class); @Test - @WithResource(name = "hazelcast.xml", content = """ - - default-instance - - - - - - - - - """) - void hibernate2ndLevelCacheWithJCacheAndHazelcast() { - String cachingProviderFqn = HazelcastServerCachingProvider.class.getName(); - String configLocation = "classpath:hazelcast.xml"; + void hibernate2ndLevelCacheWithJCacheAndEhCache() { + String cachingProviderFqn = EhcacheCachingProvider.class.getName(); this.contextRunner .withPropertyValues("spring.cache.type=jcache", "spring.cache.jcache.provider=" + cachingProviderFqn, - "spring.cache.jcache.config=" + configLocation, "spring.jpa.properties.hibernate.cache.region.factory_class=jcache", - "spring.jpa.properties.hibernate.cache.provider=" + cachingProviderFqn, - "spring.jpa.properties.hibernate.javax.cache.uri=" + configLocation) + "spring.jpa.properties.hibernate.cache.provider=" + cachingProviderFqn) .run((context) -> assertThat(context).hasNotFailed()); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaAutoConfigurationTests.java index f25bfc362d44..d64246044c3b 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaAutoConfigurationTests.java @@ -80,6 +80,8 @@ import org.springframework.context.annotation.Configuration; import org.springframework.context.event.ContextRefreshedEvent; import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.support.SQLExceptionTranslator; +import org.springframework.jdbc.support.SQLStateSQLExceptionTranslator; import org.springframework.orm.jpa.JpaTransactionManager; import org.springframework.orm.jpa.JpaVendorAdapter; import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean; @@ -234,6 +236,24 @@ void hibernateDialectIsNotSetByDefault() { (adapter) -> assertThat(adapter.getJpaPropertyMap()).doesNotContainKeys("hibernate.dialect"))); } + @Test + void shouldConfigureHibernateJpaDialectWithSqlExceptionTranslatorIfPresent() { + SQLStateSQLExceptionTranslator sqlExceptionTranslator = new SQLStateSQLExceptionTranslator(); + contextRunner().withBean(SQLStateSQLExceptionTranslator.class, () -> sqlExceptionTranslator) + .run(assertJpaVendorAdapter((adapter) -> assertThat(adapter.getJpaDialect()) + .hasFieldOrPropertyWithValue("jdbcExceptionTranslator", sqlExceptionTranslator))); + } + + @Test + void shouldNotConfigureHibernateJpaDialectWithSqlExceptionTranslatorIfNotUnique() { + SQLStateSQLExceptionTranslator sqlExceptionTranslator1 = new SQLStateSQLExceptionTranslator(); + SQLStateSQLExceptionTranslator sqlExceptionTranslator2 = new SQLStateSQLExceptionTranslator(); + contextRunner().withBean("sqlExceptionTranslator1", SQLExceptionTranslator.class, () -> sqlExceptionTranslator1) + .withBean("sqlExceptionTranslator2", SQLExceptionTranslator.class, () -> sqlExceptionTranslator2) + .run(assertJpaVendorAdapter((adapter) -> assertThat(adapter.getJpaDialect()) + .hasFieldOrPropertyWithValue("jdbcExceptionTranslator", null))); + } + @Test void hibernateDialectIsSetWhenDatabaseIsSet() { contextRunner().withPropertyValues("spring.jpa.database=H2") diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/pulsar/PulsarPropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/pulsar/PulsarPropertiesTests.java index 72fbdd0e73f4..cfaff0ccd02d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/pulsar/PulsarPropertiesTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/pulsar/PulsarPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -226,7 +226,7 @@ void bindWhenNoSchemaThrowsException() { map.put("spring.pulsar.defaults.type-mappings[0].schema-info.message-key-type", String.class.getName()); assertThatExceptionOfType(BindException.class).isThrownBy(() -> bindProperties(map)) .havingRootCause() - .withMessageContaining("schemaType must not be null"); + .withMessageContaining("'schemaType' must not be null"); } @Test @@ -236,7 +236,7 @@ void bindWhenSchemaTypeNoneThrowsException() { map.put("spring.pulsar.defaults.type-mappings[0].schema-info.schema-type", "NONE"); assertThatExceptionOfType(BindException.class).isThrownBy(() -> bindProperties(map)) .havingRootCause() - .withMessageContaining("schemaType 'NONE' not supported"); + .withMessageContaining("'schemaType' must not be NONE"); } @Test @@ -247,7 +247,7 @@ void bindWhenMessageKeyTypeSetOnNonKeyValueSchemaThrowsException() { map.put("spring.pulsar.defaults.type-mappings[0].schema-info.message-key-type", String.class.getName()); assertThatExceptionOfType(BindException.class).isThrownBy(() -> bindProperties(map)) .havingRootCause() - .withMessageContaining("messageKeyType can only be set when schemaType is KEY_VALUE"); + .withMessageContaining("'messageKeyType' can only be set when 'schemaType' is KEY_VALUE"); } record TestMessage(String value) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/quartz/QuartzAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/quartz/QuartzAutoConfigurationTests.java index 23aabcf57768..1b97cc0c4c1d 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/quartz/QuartzAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/quartz/QuartzAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -574,7 +574,7 @@ DataSourceScriptDatabaseInitializer customInitializer(DataSource dataSource) { static class ComponentThatUsesScheduler { ComponentThatUsesScheduler(Scheduler scheduler) { - Assert.notNull(scheduler, "Scheduler must not be null"); + Assert.notNull(scheduler, "'scheduler' must not be null"); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcAutoConfigurationTests.java index fb4f3eb4f107..5ec13b5792ea 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcAutoConfigurationTests.java @@ -84,8 +84,8 @@ void configureWithUrlAndPoolPropertiesApplyProperties() { this.contextRunner .withPropertyValues("spring.r2dbc.url:r2dbc:h2:mem:///" + randomDatabaseName(), "spring.r2dbc.pool.max-size=15", "spring.r2dbc.pool.max-acquire-time=3m", - "spring.r2dbc.pool.min-idle=1", "spring.r2dbc.pool.max-validation-time=1s", - "spring.r2dbc.pool.initial-size=0") + "spring.r2dbc.pool.acquire-retry=5", "spring.r2dbc.pool.min-idle=1", + "spring.r2dbc.pool.max-validation-time=1s", "spring.r2dbc.pool.initial-size=0") .run((context) -> { assertThat(context).hasSingleBean(ConnectionFactory.class) .hasSingleBean(ConnectionPool.class) @@ -98,6 +98,10 @@ void configureWithUrlAndPoolPropertiesApplyProperties() { assertThat(poolMetrics.getMaxAllocatedSize()).isEqualTo(15); assertThat(connectionPool).hasFieldOrPropertyWithValue("maxAcquireTime", Duration.ofMinutes(3)); assertThat(connectionPool).hasFieldOrPropertyWithValue("maxValidationTime", Duration.ofSeconds(1)); + assertThat(connectionPool).extracting("create").satisfies((mono) -> { + assertThat(mono.getClass().getName()).endsWith("MonoRetry"); + assertThat(mono).hasFieldOrPropertyWithValue("times", 5L); + }); } finally { connectionPool.close().block(); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/jpa/City.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/jpa/City.java new file mode 100644 index 000000000000..dbc95af5fb46 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/jpa/City.java @@ -0,0 +1,78 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.security.jpa; + +import java.io.Serializable; + +import jakarta.persistence.Column; +import jakarta.persistence.Entity; +import jakarta.persistence.GeneratedValue; +import jakarta.persistence.Id; + +@Entity +public class City implements Serializable { + + private static final long serialVersionUID = 1L; + + @Id + @GeneratedValue + private Long id; + + @Column(nullable = false) + private String name; + + @Column(nullable = false) + private String state; + + @Column(nullable = false) + private String country; + + @Column(nullable = false) + private String map; + + protected City() { + } + + public City(String name, String state, String country, String map) { + this.name = name; + this.state = state; + this.country = country; + this.map = map; + } + + public String getName() { + return this.name; + } + + public String getState() { + return this.state; + } + + public String getCountry() { + return this.country; + } + + public String getMap() { + return this.map; + } + + @Override + public String toString() { + return getName() + "," + getState() + "," + getCountry(); + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/reactive/ReactiveUserDetailsServiceAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/reactive/ReactiveUserDetailsServiceAutoConfigurationTests.java index 1b673b6dc962..41e7d556a7b8 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/reactive/ReactiveUserDetailsServiceAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/reactive/ReactiveUserDetailsServiceAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,6 @@ import org.springframework.security.core.userdetails.UserDetails; import org.springframework.security.crypto.password.PasswordEncoder; import org.springframework.security.oauth2.client.registration.ClientRegistrationRepository; -import org.springframework.security.oauth2.jwt.ReactiveJwtDecoder; import org.springframework.security.oauth2.server.resource.introspection.ReactiveOpaqueTokenIntrospector; import static org.assertj.core.api.Assertions.assertThat; @@ -108,14 +107,6 @@ void doesNotConfigureDefaultUserIfAuthenticationManagerResolverAvailable() { .doesNotHaveBean(ReactiveUserDetailsService.class)); } - @Test - void doesNotConfigureDefaultUserIfResourceServerWithJWTIsUsed() { - this.contextRunner.withUserConfiguration(JwtDecoderConfiguration.class).run((context) -> { - assertThat(context).hasSingleBean(ReactiveJwtDecoder.class); - assertThat(context).doesNotHaveBean(ReactiveUserDetailsService.class); - }); - } - @Test void doesNotConfigureDefaultUserIfResourceServerIsPresent() { this.contextRunner.run((context) -> assertThat(context).doesNotHaveBean(ReactiveUserDetailsService.class)); @@ -232,24 +223,4 @@ PasswordEncoder passwordEncoder() { } - @Configuration(proxyBeanMethods = false) - static class JwtDecoderConfiguration { - - @Bean - ReactiveJwtDecoder jwtDecoder() { - return mock(ReactiveJwtDecoder.class); - } - - } - - @Configuration(proxyBeanMethods = false) - static class ReactiveOpaqueTokenIntrospectorConfiguration { - - @Bean - ReactiveOpaqueTokenIntrospector introspectionClient() { - return mock(ReactiveOpaqueTokenIntrospector.class); - } - - } - } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/reactive/StaticResourceRequestTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/reactive/StaticResourceRequestTests.java index db501b62ae49..0b2f0957d360 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/reactive/StaticResourceRequestTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/reactive/StaticResourceRequestTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -78,13 +78,13 @@ void atLocationShouldMatchLocation() { @Test void atLocationsFromSetWhenSetIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> this.resourceRequest.at(null)) - .withMessageContaining("Locations must not be null"); + .withMessageContaining("'locations' must not be null"); } @Test void excludeFromSetWhenSetIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> this.resourceRequest.atCommonLocations().excluding(null)) - .withMessageContaining("Locations must not be null"); + .withMessageContaining("'locations' must not be null"); } private RequestMatcherAssert assertMatcher(ServerWebExchangeMatcher matcher) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/SecurityAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/SecurityAutoConfigurationTests.java index 3a26ce94b7bc..43aea7e4a11a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/SecurityAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/SecurityAutoConfigurationTests.java @@ -32,7 +32,7 @@ import org.springframework.boot.autoconfigure.context.PropertyPlaceholderAutoConfiguration; import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; import org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration; -import org.springframework.boot.autoconfigure.orm.jpa.test.City; +import org.springframework.boot.autoconfigure.security.jpa.City; import org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfiguration; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.boot.context.properties.ConfigurationPropertiesBinding; diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/StaticResourceRequestTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/StaticResourceRequestTests.java index e99120b3eed3..7f2fe7869d01 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/StaticResourceRequestTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/StaticResourceRequestTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -87,13 +87,13 @@ void atLocationWhenHasServletPathShouldMatchLocation() { @Test void atLocationsFromSetWhenSetIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> this.resourceRequest.at(null)) - .withMessageContaining("Locations must not be null"); + .withMessageContaining("'locations' must not be null"); } @Test void excludeFromSetWhenSetIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> this.resourceRequest.atCommonLocations().excluding(null)) - .withMessageContaining("Locations must not be null"); + .withMessageContaining("'locations' must not be null"); } private RequestMatcherAssert assertMatcher(RequestMatcher matcher) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/UserDetailsServiceAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/UserDetailsServiceAutoConfigurationTests.java index 6c4ee3c918ec..086f6505be83 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/UserDetailsServiceAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/servlet/UserDetailsServiceAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,19 +18,30 @@ import java.util.Collections; import java.util.function.Function; +import java.util.function.Predicate; +import java.util.stream.Stream; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.ExtendWith; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.EnumSource; import org.springframework.boot.autoconfigure.AutoConfigurations; +import org.springframework.boot.autoconfigure.condition.ConditionEvaluationReport; +import org.springframework.boot.autoconfigure.condition.ConditionEvaluationReport.ConditionAndOutcome; +import org.springframework.boot.autoconfigure.condition.ConditionEvaluationReport.ConditionAndOutcomes; +import org.springframework.boot.autoconfigure.condition.ConditionOutcome; import org.springframework.boot.autoconfigure.security.SecurityProperties; +import org.springframework.boot.autoconfigure.security.servlet.UserDetailsServiceAutoConfiguration.MissingAlternativeOrUserPropertiesConfigured; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.boot.test.context.FilteredClassLoader; +import org.springframework.boot.test.context.runner.AbstractApplicationContextRunner; import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner; import org.springframework.boot.test.context.runner.WebApplicationContextRunner; import org.springframework.boot.test.system.CapturedOutput; import org.springframework.boot.test.system.OutputCaptureExtension; +import org.springframework.context.ConfigurableApplicationContext; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Import; @@ -70,7 +81,7 @@ class UserDetailsServiceAutoConfigurationTests { @Test void shouldSupplyUserDetailsServiceInServletApp() { - this.contextRunner.with(AuthenticationExclude.servletApp()) + this.contextRunner.with(AlternativeFormOfAuthentication.nonPresent()) .run((context) -> assertThat(context).hasSingleBean(UserDetailsService.class)); } @@ -78,7 +89,7 @@ void shouldSupplyUserDetailsServiceInServletApp() { void shouldNotSupplyUserDetailsServiceInReactiveApp() { new ReactiveWebApplicationContextRunner().withUserConfiguration(TestSecurityConfiguration.class) .withConfiguration(AutoConfigurations.of(UserDetailsServiceAutoConfiguration.class)) - .with(AuthenticationExclude.reactiveApp()) + .with(AlternativeFormOfAuthentication.nonPresent()) .run((context) -> assertThat(context).doesNotHaveBean(UserDetailsService.class)); } @@ -86,13 +97,14 @@ void shouldNotSupplyUserDetailsServiceInReactiveApp() { void shouldNotSupplyUserDetailsServiceInNonWebApp() { new ApplicationContextRunner().withUserConfiguration(TestSecurityConfiguration.class) .withConfiguration(AutoConfigurations.of(UserDetailsServiceAutoConfiguration.class)) - .with(AuthenticationExclude.noWebApp()) + .with(AlternativeFormOfAuthentication.nonPresent()) .run((context) -> assertThat(context).doesNotHaveBean(UserDetailsService.class)); } @Test void testDefaultUsernamePassword(CapturedOutput output) { - this.contextRunner.with(AuthenticationExclude.servletApp()).run((context) -> { + this.contextRunner.with(AlternativeFormOfAuthentication.nonPresent()).run((context) -> { + assertThat(outcomeOfMissingAlternativeCondition(context).isMatch()).isTrue(); UserDetailsService manager = context.getBean(UserDetailsService.class); assertThat(output).contains("Using generated security password:"); assertThat(manager.loadUserByUsername("user")).isNotNull(); @@ -101,60 +113,68 @@ void testDefaultUsernamePassword(CapturedOutput output) { @Test void defaultUserNotCreatedIfAuthenticationManagerBeanPresent(CapturedOutput output) { - this.contextRunner.withUserConfiguration(TestAuthenticationManagerConfiguration.class).run((context) -> { - AuthenticationManager manager = context.getBean(AuthenticationManager.class); - assertThat(manager) - .isEqualTo(context.getBean(TestAuthenticationManagerConfiguration.class).authenticationManager); - assertThat(output).doesNotContain("Using generated security password: "); - TestingAuthenticationToken token = new TestingAuthenticationToken("foo", "bar"); - assertThat(manager.authenticate(token)).isNotNull(); - }); + this.contextRunner.with(AlternativeFormOfAuthentication.nonPresent()) + .withUserConfiguration(TestAuthenticationManagerConfiguration.class) + .run((context) -> { + assertThat(outcomeOfMissingAlternativeCondition(context).isMatch()).isTrue(); + AuthenticationManager manager = context.getBean(AuthenticationManager.class); + assertThat(manager) + .isEqualTo(context.getBean(TestAuthenticationManagerConfiguration.class).authenticationManager); + assertThat(output).doesNotContain("Using generated security password: "); + TestingAuthenticationToken token = new TestingAuthenticationToken("foo", "bar"); + assertThat(manager.authenticate(token)).isNotNull(); + }); } @Test void defaultUserNotCreatedIfAuthenticationManagerResolverBeanPresent(CapturedOutput output) { - this.contextRunner.withUserConfiguration(TestAuthenticationManagerResolverConfiguration.class) - .run((context) -> assertThat(output).doesNotContain("Using generated security password: ")); + this.contextRunner.with(AlternativeFormOfAuthentication.nonPresent()) + .withUserConfiguration(TestAuthenticationManagerResolverConfiguration.class) + .run((context) -> { + assertThat(outcomeOfMissingAlternativeCondition(context).isMatch()).isTrue(); + assertThat(output).doesNotContain("Using generated security password: "); + }); } @Test void defaultUserNotCreatedIfUserDetailsServiceBeanPresent(CapturedOutput output) { - this.contextRunner.withUserConfiguration(TestUserDetailsServiceConfiguration.class).run((context) -> { - UserDetailsService userDetailsService = context.getBean(UserDetailsService.class); - assertThat(output).doesNotContain("Using generated security password: "); - assertThat(userDetailsService.loadUserByUsername("foo")).isNotNull(); - }); + this.contextRunner.with(AlternativeFormOfAuthentication.nonPresent()) + .withUserConfiguration(TestUserDetailsServiceConfiguration.class) + .run((context) -> { + assertThat(outcomeOfMissingAlternativeCondition(context).isMatch()).isTrue(); + UserDetailsService userDetailsService = context.getBean(UserDetailsService.class); + assertThat(output).doesNotContain("Using generated security password: "); + assertThat(userDetailsService.loadUserByUsername("foo")).isNotNull(); + }); } @Test void defaultUserNotCreatedIfAuthenticationProviderBeanPresent(CapturedOutput output) { - this.contextRunner.withUserConfiguration(TestAuthenticationProviderConfiguration.class).run((context) -> { - AuthenticationProvider provider = context.getBean(AuthenticationProvider.class); - assertThat(output).doesNotContain("Using generated security password: "); - TestingAuthenticationToken token = new TestingAuthenticationToken("foo", "bar"); - assertThat(provider.authenticate(token)).isNotNull(); - }); - } - - @Test - void defaultUserNotCreatedIfResourceServerWithOpaqueIsUsed() { - this.contextRunner.withUserConfiguration(TestConfigWithIntrospectionClient.class).run((context) -> { - assertThat(context).hasSingleBean(OpaqueTokenIntrospector.class); - assertThat(context).doesNotHaveBean(UserDetailsService.class); - }); + this.contextRunner.with(AlternativeFormOfAuthentication.nonPresent()) + .withUserConfiguration(TestAuthenticationProviderConfiguration.class) + .run((context) -> { + assertThat(outcomeOfMissingAlternativeCondition(context).isMatch()).isTrue(); + AuthenticationProvider provider = context.getBean(AuthenticationProvider.class); + assertThat(output).doesNotContain("Using generated security password: "); + TestingAuthenticationToken token = new TestingAuthenticationToken("foo", "bar"); + assertThat(provider.authenticate(token)).isNotNull(); + }); } @Test - void defaultUserNotCreatedIfResourceServerWithJWTIsUsed() { - this.contextRunner.withUserConfiguration(TestConfigWithJwtDecoder.class).run((context) -> { - assertThat(context).hasSingleBean(JwtDecoder.class); - assertThat(context).doesNotHaveBean(UserDetailsService.class); - }); + void defaultUserNotCreatedIfJwtDecoderBeanPresent() { + this.contextRunner.with(AlternativeFormOfAuthentication.nonPresent()) + .withUserConfiguration(TestConfigWithJwtDecoder.class) + .run((context) -> { + assertThat(outcomeOfMissingAlternativeCondition(context).isMatch()).isTrue(); + assertThat(context).hasSingleBean(JwtDecoder.class); + assertThat(context).doesNotHaveBean(UserDetailsService.class); + }); } @Test void userDetailsServiceWhenPasswordEncoderAbsentAndDefaultPassword() { - this.contextRunner.with(AuthenticationExclude.servletApp()) + this.contextRunner.with(AlternativeFormOfAuthentication.nonPresent()) .withUserConfiguration(TestSecurityConfiguration.class) .run(((context) -> { InMemoryUserDetailsManager userDetailsService = context.getBean(InMemoryUserDetailsManager.class); @@ -179,49 +199,33 @@ void userDetailsServiceWhenPasswordEncoderBeanPresent() { testPasswordEncoding(TestConfigWithPasswordEncoder.class, "secret", "secret"); } - @Test - void userDetailsServiceWhenClientRegistrationRepositoryPresent() { - this.contextRunner - .withClassLoader( - new FilteredClassLoader(OpaqueTokenIntrospector.class, RelyingPartyRegistrationRepository.class)) - .run(((context) -> assertThat(context).doesNotHaveBean(InMemoryUserDetailsManager.class))); - } - - @Test - void userDetailsServiceWhenOpaqueTokenIntrospectorPresent() { - this.contextRunner - .withClassLoader(new FilteredClassLoader(ClientRegistrationRepository.class, - RelyingPartyRegistrationRepository.class)) - .run(((context) -> assertThat(context).doesNotHaveBean(InMemoryUserDetailsManager.class))); - } - - @Test - void userDetailsServiceWhenRelyingPartyRegistrationRepositoryPresent() { - this.contextRunner - .withClassLoader(new FilteredClassLoader(ClientRegistrationRepository.class, OpaqueTokenIntrospector.class)) - .run(((context) -> assertThat(context).doesNotHaveBean(InMemoryUserDetailsManager.class))); + @ParameterizedTest + @EnumSource + void whenClassOfAlternativeIsPresentUserDetailsServiceBacksOff(AlternativeFormOfAuthentication alternative) { + this.contextRunner.with(alternative.present()) + .run((context) -> assertThat(context).doesNotHaveBean(InMemoryUserDetailsManager.class)); } - @Test - void userDetailsServiceWhenRelyingPartyRegistrationRepositoryPresentAndUsernameConfigured() { - this.contextRunner - .withClassLoader(new FilteredClassLoader(ClientRegistrationRepository.class, OpaqueTokenIntrospector.class)) + @ParameterizedTest + @EnumSource + void whenAlternativeIsPresentAndUsernameIsConfiguredThenUserDetailsServiceIsAutoConfigured( + AlternativeFormOfAuthentication alternative) { + this.contextRunner.with(alternative.present()) .withPropertyValues("spring.security.user.name=alice") .run(((context) -> assertThat(context).hasSingleBean(InMemoryUserDetailsManager.class))); } - @Test - void userDetailsServiceWhenRelyingPartyRegistrationRepositoryPresentAndPasswordConfigured() { - this.contextRunner - .withClassLoader(new FilteredClassLoader(ClientRegistrationRepository.class, OpaqueTokenIntrospector.class)) + @ParameterizedTest + @EnumSource + void whenAlternativeIsPresentAndPasswordIsConfiguredThenUserDetailsServiceIsAutoConfigured( + AlternativeFormOfAuthentication alternative) { + this.contextRunner.with(alternative.present()) .withPropertyValues("spring.security.user.password=secret") .run(((context) -> assertThat(context).hasSingleBean(InMemoryUserDetailsManager.class))); } private void testPasswordEncoding(Class configClass, String providedPassword, String expectedPassword) { - this.contextRunner.with(AuthenticationExclude.servletApp()) - .withClassLoader(new FilteredClassLoader(ClientRegistrationRepository.class, OpaqueTokenIntrospector.class, - RelyingPartyRegistrationRepository.class)) + this.contextRunner.with(AlternativeFormOfAuthentication.nonPresent()) .withUserConfiguration(configClass) .withPropertyValues("spring.security.user.password=" + providedPassword) .run(((context) -> { @@ -231,24 +235,16 @@ private void testPasswordEncoding(Class configClass, String providedPassword, })); } - private static final class AuthenticationExclude { - - private static final FilteredClassLoader filteredClassLoader = new FilteredClassLoader( - ClientRegistrationRepository.class, OpaqueTokenIntrospector.class, - RelyingPartyRegistrationRepository.class); - - static Function servletApp() { - return (contextRunner) -> contextRunner.withClassLoader(filteredClassLoader); + private ConditionOutcome outcomeOfMissingAlternativeCondition(ConfigurableApplicationContext context) { + ConditionAndOutcomes conditionAndOutcomes = ConditionEvaluationReport.get(context.getBeanFactory()) + .getConditionAndOutcomesBySource() + .get(UserDetailsServiceAutoConfiguration.class.getName()); + for (ConditionAndOutcome conditionAndOutcome : conditionAndOutcomes) { + if (conditionAndOutcome.getCondition() instanceof MissingAlternativeOrUserPropertiesConfigured) { + return conditionAndOutcome.getOutcome(); + } } - - static Function reactiveApp() { - return (contextRunner) -> contextRunner.withClassLoader(filteredClassLoader); - } - - static Function noWebApp() { - return (contextRunner) -> contextRunner.withClassLoader(filteredClassLoader); - } - + return null; } @Configuration(proxyBeanMethods = false) @@ -346,4 +342,41 @@ AuthenticationManagerResolver authenticationManagerResolver() { } + private enum AlternativeFormOfAuthentication { + + CLIENT_REGISTRATION_REPOSITORY(ClientRegistrationRepository.class), + + OPAQUE_TOKEN_INTROSPECTOR(OpaqueTokenIntrospector.class), + + RELYING_PARTY_REGISTRATION_REPOSITORY(RelyingPartyRegistrationRepository.class); + + private final Class type; + + AlternativeFormOfAuthentication(Class type) { + this.type = type; + } + + private Class getType() { + return this.type; + } + + @SuppressWarnings("unchecked") + private > Function present() { + return (contextRunner) -> (T) contextRunner + .withClassLoader(new FilteredClassLoader(Stream.of(AlternativeFormOfAuthentication.values()) + .filter(Predicate.not(this::equals)) + .map(AlternativeFormOfAuthentication::getType) + .toArray(Class[]::new))); + } + + @SuppressWarnings("unchecked") + private static > Function nonPresent() { + return (contextRunner) -> (T) contextRunner + .withClassLoader(new FilteredClassLoader(Stream.of(AlternativeFormOfAuthentication.values()) + .map(AlternativeFormOfAuthentication::getType) + .toArray(Class[]::new))); + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/session/SessionAutoConfigurationEarlyInitializationIntegrationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/session/SessionAutoConfigurationEarlyInitializationIntegrationTests.java index da11c9381ca5..5b16c33516f2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/session/SessionAutoConfigurationEarlyInitializationIntegrationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/session/SessionAutoConfigurationEarlyInitializationIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -57,7 +57,7 @@ static class TestConfiguration { @Bean MapSessionRepository mapSessionRepository(ConfigurableApplicationContext context) { - Assert.isTrue(context.getBeanFactory().isConfigurationFrozen(), "Context should be frozen"); + Assert.isTrue(context.getBeanFactory().isConfigurationFrozen(), "'context' should be frozen"); return new MapSessionRepository(new LinkedHashMap<>()); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/session/SessionAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/session/SessionAutoConfigurationTests.java index 392ec235aeac..aa79e7737462 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/session/SessionAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/session/SessionAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -170,6 +170,16 @@ void sessionCookieConfigurationIsAppliedToAutoConfiguredCookieSerializer() { }); } + @Test + void sessionCookieSameSiteOmittedIsAppliedToAutoConfiguredCookieSerializer() { + this.contextRunner.withUserConfiguration(SessionRepositoryConfiguration.class) + .withPropertyValues("server.servlet.session.cookie.sameSite=omitted") + .run((context) -> { + DefaultCookieSerializer cookieSerializer = context.getBean(DefaultCookieSerializer.class); + assertThat(cookieSerializer).hasFieldOrPropertyWithValue("sameSite", null); + }); + } + @Test void autoConfiguredCookieSerializerIsUsedBySessionRepositoryFilter() { this.contextRunner.withUserConfiguration(SessionRepositoryConfiguration.class) diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/sql/init/OnDatabaseInitializationConditionTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/sql/init/OnDatabaseInitializationConditionTests.java index fd44e19a449f..b05a0bb4adbc 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/sql/init/OnDatabaseInitializationConditionTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/sql/init/OnDatabaseInitializationConditionTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,7 +36,7 @@ class OnDatabaseInitializationConditionTests { @Test void getMatchOutcomeWithPropertyNoSetMatches() { - OnDatabaseInitializationCondition condition = new OnDatabaseInitializationCondition("Test", "test.init-mode"); + OnDatabaseInitializationCondition condition = new OnTestDatabaseInitializationCondition("test.init-mode"); ConditionOutcome outcome = condition .getMatchOutcome(mockConditionContext(TestPropertyValues.of("test.another", "noise")), null); assertThat(outcome.isMatch()).isTrue(); @@ -44,7 +44,7 @@ void getMatchOutcomeWithPropertyNoSetMatches() { @Test void getMatchOutcomeWithPropertySetToAlwaysMatches() { - OnDatabaseInitializationCondition condition = new OnDatabaseInitializationCondition("Test", "test.init-mode"); + OnDatabaseInitializationCondition condition = new OnTestDatabaseInitializationCondition("test.init-mode"); ConditionOutcome outcome = condition .getMatchOutcome(mockConditionContext(TestPropertyValues.of("test.init-mode=always")), null); assertThat(outcome.isMatch()).isTrue(); @@ -52,7 +52,7 @@ void getMatchOutcomeWithPropertySetToAlwaysMatches() { @Test void getMatchOutcomeWithPropertySetToEmbeddedMatches() { - OnDatabaseInitializationCondition condition = new OnDatabaseInitializationCondition("Test", "test.init-mode"); + OnDatabaseInitializationCondition condition = new OnTestDatabaseInitializationCondition("test.init-mode"); ConditionOutcome outcome = condition .getMatchOutcome(mockConditionContext(TestPropertyValues.of("test.init-mode=embedded")), null); assertThat(outcome.isMatch()).isTrue(); @@ -60,7 +60,7 @@ void getMatchOutcomeWithPropertySetToEmbeddedMatches() { @Test void getMatchOutcomeWithPropertySetToNeverDoesNotMatch() { - OnDatabaseInitializationCondition condition = new OnDatabaseInitializationCondition("Test", "test.init-mode"); + OnDatabaseInitializationCondition condition = new OnTestDatabaseInitializationCondition("test.init-mode"); ConditionOutcome outcome = condition .getMatchOutcome(mockConditionContext(TestPropertyValues.of("test.init-mode=never")), null); assertThat(outcome.isMatch()).isFalse(); @@ -68,7 +68,7 @@ void getMatchOutcomeWithPropertySetToNeverDoesNotMatch() { @Test void getMatchOutcomeWithPropertySetToEmptyStringIsIgnored() { - OnDatabaseInitializationCondition condition = new OnDatabaseInitializationCondition("Test", "test.init-mode"); + OnDatabaseInitializationCondition condition = new OnTestDatabaseInitializationCondition("test.init-mode"); ConditionOutcome outcome = condition .getMatchOutcome(mockConditionContext(TestPropertyValues.of("test.init-mode")), null); assertThat(outcome.isMatch()).isTrue(); @@ -76,7 +76,7 @@ void getMatchOutcomeWithPropertySetToEmptyStringIsIgnored() { @Test void getMatchOutcomeWithMultiplePropertiesUsesFirstSet() { - OnDatabaseInitializationCondition condition = new OnDatabaseInitializationCondition("Test", "test.init-mode", + OnDatabaseInitializationCondition condition = new OnTestDatabaseInitializationCondition("test.init-mode", "test.schema-mode", "test.init-schema-mode"); ConditionOutcome outcome = condition .getMatchOutcome(mockConditionContext(TestPropertyValues.of("test.init-schema-mode=embedded")), null); @@ -86,7 +86,7 @@ void getMatchOutcomeWithMultiplePropertiesUsesFirstSet() { @Test void getMatchOutcomeHasDedicatedDescription() { - OnDatabaseInitializationCondition condition = new OnDatabaseInitializationCondition("Test", "test.init-mode"); + OnDatabaseInitializationCondition condition = new OnTestDatabaseInitializationCondition("test.init-mode"); ConditionOutcome outcome = condition .getMatchOutcome(mockConditionContext(TestPropertyValues.of("test.init-mode=embedded")), null); assertThat(outcome.getMessage()).isEqualTo("TestDatabase Initialization test.init-mode is EMBEDDED"); @@ -94,7 +94,7 @@ void getMatchOutcomeHasDedicatedDescription() { @Test void getMatchOutcomeHasWhenPropertyIsNotSetHasDefaultDescription() { - OnDatabaseInitializationCondition condition = new OnDatabaseInitializationCondition("Test", "test.init-mode"); + OnDatabaseInitializationCondition condition = new OnTestDatabaseInitializationCondition("test.init-mode"); ConditionOutcome outcome = condition.getMatchOutcome(mockConditionContext(TestPropertyValues.empty()), null); assertThat(outcome.getMessage()).isEqualTo("TestDatabase Initialization default value is EMBEDDED"); } @@ -107,4 +107,12 @@ private ConditionContext mockConditionContext(TestPropertyValues propertyValues) return conditionContext; } + static class OnTestDatabaseInitializationCondition extends OnDatabaseInitializationCondition { + + OnTestDatabaseInitializationCondition(String... properties) { + super("Test", properties); + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/ssl/FileWatcherTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/ssl/FileWatcherTests.java index 7a4d4061d14e..07d651b1a75f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/ssl/FileWatcherTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/ssl/FileWatcherTests.java @@ -18,8 +18,10 @@ import java.io.IOException; import java.io.UncheckedIOException; +import java.nio.file.AccessDeniedException; import java.nio.file.Files; import java.nio.file.Path; +import java.nio.file.StandardCopyOption; import java.time.Duration; import java.util.Set; import java.util.UUID; @@ -254,6 +256,64 @@ void shouldTriggerOnConfigMapUpdates(@TempDir Path tempDir) throws Exception { } } + /** + * Updates many times K8s ConfigMap/Secret with atomic move.
+	 * .
+	 * ├── ..a72e81ff-f0e1-41d8-a19b-068d3d1d4e2f
+	 * │   ├── keystore.jks
+	 * ├── ..data -> ..a72e81ff-f0e1-41d8-a19b-068d3d1d4e2f
+	 * ├── keystore.jks -> ..data/keystore.jks
+	 * 
+ * + * After a first a ConfigMap/Secret update, this will look like:
+	 * .
+	 * ├── ..bba2a61f-ce04-4c35-93aa-e455110d4487
+	 * │   ├── keystore.jks
+	 * ├── ..data -> ..bba2a61f-ce04-4c35-93aa-e455110d4487
+	 * ├── keystore.jks -> ..data/keystore.jks
+	 * 
After a second a ConfigMap/Secret update, this will look like:
+	 * .
+	 * ├── ..134887f0-df8f-4433-b70c-7784d2a33bd1
+	 * │   ├── keystore.jks
+	 * ├── ..data -> ..134887f0-df8f-4433-b70c-7784d2a33bd1
+	 * ├── keystore.jks -> ..data/keystore.jks
+	 *
+ *

+ * When Kubernetes updates either the ConfigMap or Secret, it performs the following + * steps: + *

    + *
  • Creates a new unique directory.
  • + *
  • Writes the ConfigMap/Secret content to the newly created directory.
  • + *
  • Creates a symlink {@code ..data_tmp} pointing to the newly created + * directory.
  • + *
  • Performs an atomic rename of {@code ..data_tmp} to {@code ..data}.
  • + *
  • Deletes the old ConfigMap/Secret directory.
  • + *
+ * @param tempDir temp directory + * @throws Exception if a failure occurs + */ + @Test + void shouldTriggerOnConfigMapAtomicMoveUpdates(@TempDir Path tempDir) throws Exception { + Path configMap1 = createConfigMap(tempDir, "keystore.jks"); + Path data = Files.createSymbolicLink(tempDir.resolve("..data"), configMap1); + Files.createSymbolicLink(tempDir.resolve("keystore.jks"), data.resolve("keystore.jks")); + WaitingCallback callback = new WaitingCallback(); + this.fileWatcher.watch(Set.of(tempDir.resolve("keystore.jks")), callback); + // First update + Path configMap2 = createConfigMap(tempDir, "keystore.jks"); + Path dataTmp = Files.createSymbolicLink(tempDir.resolve("..data_tmp"), configMap2); + move(dataTmp, data); + FileSystemUtils.deleteRecursively(configMap1); + callback.expectChanges(); + callback.reset(); + // Second update + Path configMap3 = createConfigMap(tempDir, "keystore.jks"); + dataTmp = Files.createSymbolicLink(tempDir.resolve("..data_tmp"), configMap3); + move(dataTmp, data); + FileSystemUtils.deleteRecursively(configMap2); + callback.expectChanges(); + } + Path createConfigMap(Path parentDir, String secretFileName) throws IOException { Path configMapFolder = parentDir.resolve(".." + UUID.randomUUID()); Files.createDirectory(configMapFolder); @@ -262,9 +322,19 @@ Path createConfigMap(Path parentDir, String secretFileName) throws IOException { return configMapFolder; } + private void move(Path source, Path target) throws IOException { + try { + Files.move(source, target, StandardCopyOption.ATOMIC_MOVE); + } + catch (AccessDeniedException ex) { + // Windows + Files.move(source, target, StandardCopyOption.REPLACE_EXISTING); + } + } + private static final class WaitingCallback implements Runnable { - private final CountDownLatch latch = new CountDownLatch(1); + private CountDownLatch latch = new CountDownLatch(1); volatile boolean changed = false; @@ -292,6 +362,11 @@ void waitForChanges(boolean fail) throws InterruptedException { } } + void reset() { + this.latch = new CountDownLatch(1); + this.changed = false; + } + } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/task/TaskExecutionAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/task/TaskExecutionAutoConfigurationTests.java index f8c6f2f337fb..a5d92e2a35ef 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/task/TaskExecutionAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/task/TaskExecutionAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,6 +30,8 @@ import org.junit.jupiter.api.extension.ExtendWith; import org.springframework.beans.factory.config.BeanDefinition; +import org.springframework.beans.factory.config.BeanFactoryPostProcessor; +import org.springframework.beans.factory.support.BeanDefinitionOverrideException; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.task.SimpleAsyncTaskExecutorBuilder; import org.springframework.boot.task.ThreadPoolTaskExecutorBuilder; @@ -44,6 +46,7 @@ import org.springframework.core.task.TaskDecorator; import org.springframework.core.task.TaskExecutor; import org.springframework.scheduling.annotation.Async; +import org.springframework.scheduling.annotation.AsyncConfigurer; import org.springframework.scheduling.annotation.EnableAsync; import org.springframework.scheduling.annotation.EnableScheduling; import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; @@ -204,16 +207,58 @@ void simpleAsyncTaskExecutorBuilderUsesVirtualThreadsWhenEnabled() { @Test void taskExecutorWhenHasCustomTaskExecutorShouldBackOff() { - this.contextRunner.withUserConfiguration(CustomTaskExecutorConfig.class).run((context) -> { + this.contextRunner.withBean("customTaskExecutor", Executor.class, SyncTaskExecutor::new).run((context) -> { assertThat(context).hasSingleBean(Executor.class); assertThat(context.getBean(Executor.class)).isSameAs(context.getBean("customTaskExecutor")); }); } + @Test + void taskExecutorWhenModeIsAutoAndHasCustomTaskExecutorShouldBackOff() { + this.contextRunner.withBean("customTaskExecutor", Executor.class, SyncTaskExecutor::new) + .withPropertyValues("spring.task.execution.mode=auto") + .run((context) -> { + assertThat(context).hasSingleBean(Executor.class); + assertThat(context.getBean(Executor.class)).isSameAs(context.getBean("customTaskExecutor")); + }); + } + + @Test + void taskExecutorWhenModeIsForceAndHasCustomTaskExecutorShouldCreateApplicationTaskExecutor() { + this.contextRunner.withBean("customTaskExecutor", Executor.class, SyncTaskExecutor::new) + .withPropertyValues("spring.task.execution.mode=force") + .run((context) -> assertThat(context.getBeansOfType(Executor.class)).hasSize(2) + .containsKeys("customTaskExecutor", "applicationTaskExecutor")); + } + + @Test + void taskExecutorWhenModeIsForceAndHasCustomTaskExecutorWithReservedNameShouldThrowException() { + this.contextRunner.withBean("applicationTaskExecutor", Executor.class, SyncTaskExecutor::new) + .withPropertyValues("spring.task.execution.mode=force") + .run((context) -> assertThat(context).hasFailed() + .getFailure() + .isInstanceOf(BeanDefinitionOverrideException.class)); + } + + @Test + void taskExecutorWhenModeIsForceAndHasCustomBFPPCanRestoreTaskExecutorAlias() { + this.contextRunner.withBean("customTaskExecutor", Executor.class, SyncTaskExecutor::new) + .withPropertyValues("spring.task.execution.mode=force") + .withBean(BeanFactoryPostProcessor.class, + () -> (beanFactory) -> beanFactory.registerAlias("applicationTaskExecutor", "taskExecutor")) + .run((context) -> { + assertThat(context.getBeansOfType(Executor.class)).hasSize(2) + .containsKeys("customTaskExecutor", "applicationTaskExecutor"); + assertThat(context).hasBean("taskExecutor"); + assertThat(context.getBean("taskExecutor")).isSameAs(context.getBean("applicationTaskExecutor")); + + }); + } + @Test @EnabledForJreRange(min = JRE.JAVA_21) void whenVirtualThreadsAreEnabledAndCustomTaskExecutorIsDefinedThenSimpleAsyncTaskExecutorThatUsesVirtualThreadsBacksOff() { - this.contextRunner.withUserConfiguration(CustomTaskExecutorConfig.class) + this.contextRunner.withBean("customTaskExecutor", Executor.class, SyncTaskExecutor::new) .withPropertyValues("spring.threads.virtual.enabled=true") .run((context) -> { assertThat(context).hasSingleBean(Executor.class); @@ -223,25 +268,118 @@ void whenVirtualThreadsAreEnabledAndCustomTaskExecutorIsDefinedThenSimpleAsyncTa @Test void enableAsyncUsesAutoConfiguredOneByDefault() { - this.contextRunner.withPropertyValues("spring.task.execution.thread-name-prefix=task-test-") + this.contextRunner.withPropertyValues("spring.task.execution.thread-name-prefix=auto-task-") .withUserConfiguration(AsyncConfiguration.class, TestBean.class) .run((context) -> { + assertThat(context).hasSingleBean(AsyncConfigurer.class); assertThat(context).hasSingleBean(TaskExecutor.class); TestBean bean = context.getBean(TestBean.class); String text = bean.echo("something").get(); - assertThat(text).contains("task-test-").contains("something"); + assertThat(text).contains("auto-task-").contains("something"); + }); + } + + @Test + void enableAsyncUsesCustomExecutorIfPresent() { + this.contextRunner.withPropertyValues("spring.task.execution.thread-name-prefix=auto-task-") + .withBean("customTaskExecutor", Executor.class, () -> createCustomAsyncExecutor("custom-task-")) + .withUserConfiguration(AsyncConfiguration.class, TestBean.class) + .run((context) -> { + assertThat(context).doesNotHaveBean(AsyncConfigurer.class); + assertThat(context).hasSingleBean(Executor.class); + TestBean bean = context.getBean(TestBean.class); + String text = bean.echo("something").get(); + assertThat(text).contains("custom-task-").contains("something"); + }); + } + + @Test + void enableAsyncUsesAutoConfiguredExecutorWhenModeIsForceAndHasCustomTaskExecutor() { + this.contextRunner + .withPropertyValues("spring.task.execution.thread-name-prefix=auto-task-", + "spring.task.execution.mode=force") + .withBean("customTaskExecutor", Executor.class, () -> createCustomAsyncExecutor("custom-task-")) + .withUserConfiguration(AsyncConfiguration.class, TestBean.class) + .run((context) -> { + assertThat(context).hasSingleBean(AsyncConfigurer.class); + assertThat(context.getBeansOfType(Executor.class)).hasSize(2); + TestBean bean = context.getBean(TestBean.class); + String text = bean.echo("something").get(); + assertThat(text).contains("auto-task-").contains("something"); }); } + @Test + void enableAsyncUsesAutoConfiguredExecutorWhenModeIsForceAndHasCustomTaskExecutorWithReservedName() { + this.contextRunner + .withPropertyValues("spring.task.execution.thread-name-prefix=auto-task-", + "spring.task.execution.mode=force") + .withBean("taskExecutor", Executor.class, () -> createCustomAsyncExecutor("custom-task-")) + .withUserConfiguration(AsyncConfiguration.class, TestBean.class) + .run((context) -> { + assertThat(context).hasSingleBean(AsyncConfigurer.class); + assertThat(context.getBeansOfType(Executor.class)).hasSize(2); + TestBean bean = context.getBean(TestBean.class); + String text = bean.echo("something").get(); + assertThat(text).contains("auto-task-").contains("something"); + }); + } + + @Test + void enableAsyncUsesAsyncConfigurerWhenModeIsForce() { + this.contextRunner + .withPropertyValues("spring.task.execution.thread-name-prefix=auto-task-", + "spring.task.execution.mode=force") + .withBean("taskExecutor", Executor.class, () -> createCustomAsyncExecutor("custom-task-")) + .withBean("customAsyncConfigurer", AsyncConfigurer.class, () -> new AsyncConfigurer() { + @Override + public Executor getAsyncExecutor() { + return createCustomAsyncExecutor("async-task-"); + } + }) + .withUserConfiguration(AsyncConfiguration.class, TestBean.class) + .run((context) -> { + assertThat(context).hasSingleBean(AsyncConfigurer.class); + assertThat(context.getBeansOfType(Executor.class)).hasSize(2) + .containsOnlyKeys("taskExecutor", "applicationTaskExecutor"); + TestBean bean = context.getBean(TestBean.class); + String text = bean.echo("something").get(); + assertThat(text).contains("async-task-").contains("something"); + }); + } + + @Test + void enableAsyncUsesAutoConfiguredExecutorWhenModeIsForceAndHasPrimaryCustomTaskExecutor() { + this.contextRunner + .withPropertyValues("spring.task.execution.thread-name-prefix=auto-task-", + "spring.task.execution.mode=force") + .withBean("taskExecutor", Executor.class, () -> createCustomAsyncExecutor("custom-task-"), + (bd) -> bd.setPrimary(true)) + .withUserConfiguration(AsyncConfiguration.class, TestBean.class) + .run((context) -> { + assertThat(context).hasSingleBean(AsyncConfigurer.class); + assertThat(context.getBeansOfType(Executor.class)).hasSize(2); + TestBean bean = context.getBean(TestBean.class); + String text = bean.echo("something").get(); + assertThat(text).contains("auto-task-").contains("something"); + }); + } + + private Executor createCustomAsyncExecutor(String threadNamePrefix) { + SimpleAsyncTaskExecutor executor = new SimpleAsyncTaskExecutor(); + executor.setThreadNamePrefix(threadNamePrefix); + return executor; + } + @Test void enableAsyncUsesAutoConfiguredOneByDefaultEvenThoughSchedulingIsConfigured() { - this.contextRunner.withPropertyValues("spring.task.execution.thread-name-prefix=task-test-") + this.contextRunner.withPropertyValues("spring.task.execution.thread-name-prefix=auto-task-") .withConfiguration(AutoConfigurations.of(TaskSchedulingAutoConfiguration.class)) .withUserConfiguration(AsyncConfiguration.class, SchedulingConfiguration.class, TestBean.class) .run((context) -> { TestBean bean = context.getBean(TestBean.class); String text = bean.echo("something").get(); - assertThat(text).contains("task-test-").contains("something"); + assertThat(text).contains("auto-task-").contains("something"); }); } @@ -299,16 +437,6 @@ TaskDecorator mockTaskDecorator() { } - @Configuration(proxyBeanMethods = false) - static class CustomTaskExecutorConfig { - - @Bean - Executor customTaskExecutor() { - return new SyncTaskExecutor(); - } - - } - @Configuration(proxyBeanMethods = false) @EnableAsync static class AsyncConfiguration { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/task/TaskSchedulingAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/task/TaskSchedulingAutoConfigurationTests.java index 74dc49d97403..2624a5f28706 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/task/TaskSchedulingAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/task/TaskSchedulingAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,6 +41,7 @@ import org.springframework.boot.test.context.runner.ApplicationContextRunner; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.core.task.TaskDecorator; import org.springframework.core.task.TaskExecutor; import org.springframework.scheduling.TaskScheduler; import org.springframework.scheduling.annotation.EnableScheduling; @@ -50,6 +51,7 @@ import org.springframework.scheduling.config.ScheduledTaskRegistrar; import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.Mockito.mock; /** * Tests for {@link TaskSchedulingAutoConfiguration}. @@ -139,6 +141,30 @@ void simpleAsyncTaskSchedulerBuilderShouldUsePlatformThreadsByDefault() { }); } + @Test + void simpleAsyncTaskSchedulerBuilderShouldApplyTaskDecorator() { + this.contextRunner.withUserConfiguration(SchedulingConfiguration.class, TaskDecoratorConfig.class) + .run((context) -> { + assertThat(context).hasSingleBean(SimpleAsyncTaskSchedulerBuilder.class); + assertThat(context).hasSingleBean(TaskDecorator.class); + TaskDecorator taskDecorator = context.getBean(TaskDecorator.class); + SimpleAsyncTaskSchedulerBuilder builder = context.getBean(SimpleAsyncTaskSchedulerBuilder.class); + assertThat(builder).extracting("taskDecorator").isSameAs(taskDecorator); + }); + } + + @Test + void threadPoolTaskSchedulerBuilderShouldApplyTaskDecorator() { + this.contextRunner.withUserConfiguration(SchedulingConfiguration.class, TaskDecoratorConfig.class) + .run((context) -> { + assertThat(context).hasSingleBean(ThreadPoolTaskSchedulerBuilder.class); + assertThat(context).hasSingleBean(TaskDecorator.class); + TaskDecorator taskDecorator = context.getBean(TaskDecorator.class); + ThreadPoolTaskSchedulerBuilder builder = context.getBean(ThreadPoolTaskSchedulerBuilder.class); + assertThat(builder).extracting("taskDecorator").isSameAs(taskDecorator); + }); + } + @Test void simpleAsyncTaskSchedulerBuilderShouldApplyCustomizers() { SimpleAsyncTaskSchedulerCustomizer customizer = (scheduler) -> { @@ -305,4 +331,14 @@ static class TestTaskScheduler extends ThreadPoolTaskScheduler { } + @Configuration(proxyBeanMethods = false) + static class TaskDecoratorConfig { + + @Bean + TaskDecorator mockTaskDecorator() { + return mock(TaskDecorator.class); + } + + } + } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/template/TemplateAvailabilityProvidersTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/template/TemplateAvailabilityProvidersTests.java index 48cc8cb8c578..4591a1d3eca5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/template/TemplateAvailabilityProvidersTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/template/TemplateAvailabilityProvidersTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -67,7 +67,7 @@ void setup() { void createWhenApplicationContextIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TemplateAvailabilityProviders((ApplicationContext) null)) - .withMessageContaining("ClassLoader must not be null"); + .withMessageContaining("'classLoader' must not be null"); } @Test @@ -82,7 +82,7 @@ void createWhenUsingApplicationContextShouldLoadProviders() { @Test void createWhenClassLoaderIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new TemplateAvailabilityProviders((ClassLoader) null)) - .withMessageContaining("ClassLoader must not be null"); + .withMessageContaining("'classLoader' must not be null"); } @Test @@ -95,7 +95,7 @@ void createWhenUsingClassLoaderShouldLoadProviders() { void createWhenProvidersIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new TemplateAvailabilityProviders((Collection) null)) - .withMessageContaining("Providers must not be null"); + .withMessageContaining("'providers' must not be null"); } @Test @@ -108,35 +108,35 @@ void createWhenUsingProvidersShouldUseProviders() { @Test void getProviderWhenApplicationContextIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> this.providers.getProvider(this.view, null)) - .withMessageContaining("ApplicationContext must not be null"); + .withMessageContaining("'applicationContext' must not be null"); } @Test void getProviderWhenViewIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> this.providers.getProvider(null, this.environment, this.classLoader, this.resourceLoader)) - .withMessageContaining("View must not be null"); + .withMessageContaining("'view' must not be null"); } @Test void getProviderWhenEnvironmentIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> this.providers.getProvider(this.view, null, this.classLoader, this.resourceLoader)) - .withMessageContaining("Environment must not be null"); + .withMessageContaining("'environment' must not be null"); } @Test void getProviderWhenClassLoaderIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> this.providers.getProvider(this.view, this.environment, null, this.resourceLoader)) - .withMessageContaining("ClassLoader must not be null"); + .withMessageContaining("'classLoader' must not be null"); } @Test void getProviderWhenResourceLoaderIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> this.providers.getProvider(this.view, this.environment, this.classLoader, null)) - .withMessageContaining("ResourceLoader must not be null"); + .withMessageContaining("'resourceLoader' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/validation/ValidationAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/validation/ValidationAutoConfigurationTests.java index 66dc324099c3..2136c1d991fc 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/validation/ValidationAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/validation/ValidationAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -46,6 +46,7 @@ import org.springframework.validation.beanvalidation.LocalValidatorFactoryBean; import org.springframework.validation.beanvalidation.MethodValidationPostProcessor; import org.springframework.validation.beanvalidation.OptionalValidatorFactoryBean; +import org.springframework.validation.method.MethodValidationException; import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatExceptionOfType; @@ -59,6 +60,7 @@ * * @author Stephane Nicoll * @author Phillip Webb + * @author Yanming Zhou */ class ValidationAutoConfigurationTests { @@ -207,6 +209,18 @@ void validationCanBeConfiguredToUseJdkProxy() { }); } + @Test + void validationCanBeConfiguredToAdaptConstraintViolations() { + this.contextRunner.withUserConfiguration(AnotherSampleServiceConfiguration.class) + .withPropertyValues("spring.validation.method.adapt-constraint-violations=true") + .run((context) -> { + assertThat(context.getBeansOfType(Validator.class)).hasSize(1); + AnotherSampleService service = context.getBean(AnotherSampleService.class); + service.doSomething(42); + assertThatExceptionOfType(MethodValidationException.class).isThrownBy(() -> service.doSomething(2)); + }); + } + @Test @SuppressWarnings("unchecked") void userDefinedMethodValidationPostProcessorTakesPrecedence() { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/validation/ValidationPropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/validation/ValidationPropertiesTests.java new file mode 100644 index 000000000000..d507516c8011 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/validation/ValidationPropertiesTests.java @@ -0,0 +1,38 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.autoconfigure.validation; + +import org.junit.jupiter.api.Test; + +import org.springframework.validation.beanvalidation.MethodValidationPostProcessor; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link ValidationProperties}. + * + * @author Andy Wilkinson + */ +class ValidationPropertiesTests { + + @Test + void adaptConstraintViolationsPropertyDefaultMatchesPostProcessorDefault() { + assertThat(new MethodValidationPostProcessor()).extracting("adaptConstraintViolations") + .isEqualTo(new ValidationProperties().getMethod().isAdaptConstraintViolations()); + } + +} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/ServerPropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/ServerPropertiesTests.java index a7fa8a4dd642..55c41832809f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/ServerPropertiesTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/ServerPropertiesTests.java @@ -35,8 +35,6 @@ import org.eclipse.jetty.server.Server; import org.eclipse.jetty.util.thread.QueuedThreadPool; import org.junit.jupiter.api.Test; -import org.junit.jupiter.api.condition.EnabledForJreRange; -import org.junit.jupiter.api.condition.JRE; import reactor.netty.http.HttpDecoderSpec; import org.springframework.boot.autoconfigure.web.ServerProperties.Tomcat.Accesslog; @@ -203,7 +201,7 @@ void testCustomizedMimeMapping() { } @Test - void testCustomizeUriEncoding() { + void testCustomizeTomcatUriEncoding() { bind("server.tomcat.uri-encoding", "US-ASCII"); assertThat(this.properties.getTomcat().getUriEncoding()).isEqualTo(StandardCharsets.US_ASCII); } @@ -239,17 +237,23 @@ void testCustomizeTomcatKeepAliveTimeoutWithInfinite() { } @Test - void customizeMaxKeepAliveRequests() { + void testCustomizeTomcatMaxKeepAliveRequests() { bind("server.tomcat.max-keep-alive-requests", "200"); assertThat(this.properties.getTomcat().getMaxKeepAliveRequests()).isEqualTo(200); } @Test - void customizeMaxKeepAliveRequestsWithInfinite() { + void testCustomizeTomcatMaxKeepAliveRequestsWithInfinite() { bind("server.tomcat.max-keep-alive-requests", "-1"); assertThat(this.properties.getTomcat().getMaxKeepAliveRequests()).isEqualTo(-1); } + @Test + void testCustomizeTomcatMaxParameterCount() { + bind("server.tomcat.max-parameter-count", "100"); + assertThat(this.properties.getTomcat().getMaxParameterCount()).isEqualTo(100); + } + @Test void testCustomizeTomcatMinSpareThreads() { bind("server.tomcat.threads.min-spare", "10"); @@ -383,6 +387,12 @@ void tomcatMaxHttpPostSizeMatchesConnectorDefault() { .isEqualTo(getDefaultConnector().getMaxPostSize()); } + @Test + void tomcatMaxParameterCountMatchesConnectorDefault() { + assertThat(this.properties.getTomcat().getMaxParameterCount()) + .isEqualTo(getDefaultConnector().getMaxParameterCount()); + } + @Test void tomcatBackgroundProcessorDelayMatchesEngineDefault() { assertThat(this.properties.getTomcat().getBackgroundProcessorDelay()) @@ -499,14 +509,7 @@ void nettyInitialBufferSizeMatchesHttpDecoderSpecDefault() { } @Test - @EnabledForJreRange(max = JRE.JAVA_23) - void shouldDefaultAprToWhenAvailableUntilJava23() { - assertThat(this.properties.getTomcat().getUseApr()).isEqualTo(UseApr.WHEN_AVAILABLE); - } - - @Test - @EnabledForJreRange(min = JRE.JAVA_24) - void shouldDefaultAprToNeverOnJava24AndLater() { + void shouldDefaultAprToNever() { assertThat(this.properties.getTomcat().getUseApr()).isEqualTo(UseApr.NEVER); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/client/HttpMessageConvertersRestClientCustomizerTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/client/HttpMessageConvertersRestClientCustomizerTests.java index f4138baa1819..cf7c02fdbe8a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/client/HttpMessageConvertersRestClientCustomizerTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/client/HttpMessageConvertersRestClientCustomizerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -44,7 +44,7 @@ class HttpMessageConvertersRestClientCustomizerTests { void createWhenNullMessageConvertersArrayThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new HttpMessageConvertersRestClientCustomizer((HttpMessageConverter[]) null)) - .withMessage("MessageConverters must not be null"); + .withMessage("'messageConverters' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/embedded/TomcatWebServerFactoryCustomizerTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/embedded/TomcatWebServerFactoryCustomizerTests.java index f3c1ed493041..286ddd40ac2f 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/embedded/TomcatWebServerFactoryCustomizerTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/embedded/TomcatWebServerFactoryCustomizerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -194,6 +194,13 @@ void customMaxHttpRequestHeaderSize() { .isEqualTo(DataSize.ofMegabytes(10).toBytes())); } + @Test + void customMaxParameterCount() { + bind("server.tomcat.max-parameter-count=100"); + customizeAndRunServer( + (server) -> assertThat(server.getTomcat().getConnector().getMaxParameterCount()).isEqualTo(100)); + } + @Test void customMaxRequestHttpHeaderSizeIgnoredIfNegative() { bind("server.max-http-request-header-size=-1"); diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfigurationTests.java index ac9ed313b725..c7bd34d8be9a 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfigurationTests.java @@ -676,6 +676,15 @@ void customSessionCookieConfigurationShouldBeApplied() { })); } + @Test + void sessionCookieOmittedConfigurationShouldBeApplied() { + this.contextRunner.withPropertyValues("server.reactive.session.cookie.same-site:omitted") + .run(assertExchangeWithSession((exchange) -> { + List cookies = exchange.getResponse().getCookies().get("SESSION"); + assertThat(cookies).extracting(ResponseCookie::getSameSite).containsOnlyNulls(); + })); + } + @ParameterizedTest @ValueSource(classes = { ServerProperties.class, WebFluxProperties.class }) void propertiesAreNotEnabledInNonWebApplication(Class propertiesClass) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/AbstractClientHttpConnectorFactoryTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/AbstractClientHttpConnectorFactoryTests.java deleted file mode 100644 index 44f573b737b0..000000000000 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/AbstractClientHttpConnectorFactoryTests.java +++ /dev/null @@ -1,106 +0,0 @@ -/* - * Copyright 2012-2025 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.autoconfigure.web.reactive.function.client; - -import org.junit.jupiter.api.Test; - -import org.springframework.boot.ssl.SslBundle; -import org.springframework.boot.ssl.SslBundleKey; -import org.springframework.boot.ssl.jks.JksSslStoreBundle; -import org.springframework.boot.ssl.jks.JksSslStoreDetails; -import org.springframework.boot.testsupport.classpath.resources.WithPackageResources; -import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory; -import org.springframework.boot.web.server.Ssl; -import org.springframework.boot.web.server.Ssl.ClientAuth; -import org.springframework.boot.web.server.WebServer; -import org.springframework.web.reactive.function.client.WebClient; -import org.springframework.web.reactive.function.client.WebClientRequestException; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.assertThatExceptionOfType; - -/** - * Abstract base class for {@link ClientHttpConnectorFactory} tests. - * - * @author Phillip Webb - */ -abstract class AbstractClientHttpConnectorFactoryTests { - - @Test - void insecureConnection() { - TomcatServletWebServerFactory webServerFactory = new TomcatServletWebServerFactory(0); - WebServer webServer = webServerFactory.getWebServer(); - try { - webServer.start(); - int port = webServer.getPort(); - String url = "http://localhost:%s".formatted(port); - WebClient insecureWebClient = WebClient.builder() - .clientConnector(getFactory().createClientHttpConnector()) - .build(); - String insecureBody = insecureWebClient.get() - .uri(url) - .exchangeToMono((response) -> response.bodyToMono(String.class)) - .block(); - assertThat(insecureBody).contains("HTTP Status 404 – Not Found"); - } - finally { - webServer.stop(); - } - } - - @Test - @WithPackageResources("test.jks") - void secureConnection() throws Exception { - TomcatServletWebServerFactory webServerFactory = new TomcatServletWebServerFactory(0); - Ssl ssl = new Ssl(); - ssl.setClientAuth(ClientAuth.NEED); - ssl.setKeyPassword("password"); - ssl.setKeyStore("classpath:test.jks"); - ssl.setTrustStore("classpath:test.jks"); - webServerFactory.setSsl(ssl); - WebServer webServer = webServerFactory.getWebServer(); - try { - webServer.start(); - int port = webServer.getPort(); - String url = "https://localhost:%s".formatted(port); - WebClient insecureWebClient = WebClient.builder() - .clientConnector(getFactory().createClientHttpConnector()) - .build(); - assertThatExceptionOfType(WebClientRequestException.class).isThrownBy(() -> insecureWebClient.get() - .uri(url) - .exchangeToMono((response) -> response.bodyToMono(String.class)) - .block()); - JksSslStoreDetails storeDetails = JksSslStoreDetails.forLocation("classpath:test.jks"); - JksSslStoreBundle stores = new JksSslStoreBundle(storeDetails, storeDetails); - SslBundle sslBundle = SslBundle.of(stores, SslBundleKey.of("password")); - WebClient secureWebClient = WebClient.builder() - .clientConnector(getFactory().createClientHttpConnector(sslBundle)) - .build(); - String secureBody = secureWebClient.get() - .uri(url) - .exchangeToMono((response) -> response.bodyToMono(String.class)) - .block(); - assertThat(secureBody).contains("HTTP Status 404 – Not Found"); - } - finally { - webServer.stop(); - } - } - - protected abstract ClientHttpConnectorFactory getFactory(); - -} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorAutoConfigurationTests.java index 9c0d2ee1f2b3..c98f1ba32535 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,146 +16,45 @@ package org.springframework.boot.autoconfigure.web.reactive.function.client; -import org.apache.hc.client5.http.impl.async.HttpAsyncClients; import org.junit.jupiter.api.Test; -import reactor.netty.http.client.HttpClient; -import org.springframework.beans.factory.config.BeanDefinition; import org.springframework.boot.autoconfigure.AutoConfigurations; -import org.springframework.boot.test.context.FilteredClassLoader; -import org.springframework.boot.test.context.runner.ApplicationContextRunner; -import org.springframework.boot.web.reactive.function.client.WebClientCustomizer; +import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner; import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.http.client.ReactorResourceFactory; import org.springframework.http.client.reactive.ClientHttpConnector; -import org.springframework.http.client.reactive.ReactorClientHttpConnector; -import org.springframework.web.reactive.function.client.WebClient; import static org.assertj.core.api.Assertions.assertThat; -import static org.mockito.ArgumentMatchers.any; -import static org.mockito.BDDMockito.then; -import static org.mockito.Mockito.mock; /** * Tests for {@link ClientHttpConnectorAutoConfiguration} * * @author Brian Clozel */ +@SuppressWarnings("removal") class ClientHttpConnectorAutoConfigurationTests { - private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() - .withConfiguration(AutoConfigurations.of(ClientHttpConnectorAutoConfiguration.class)); - - @Test - void whenReactorIsAvailableThenReactorBeansAreDefined() { - this.contextRunner.run((context) -> { - BeanDefinition customizerDefinition = context.getBeanFactory() - .getBeanDefinition("webClientHttpConnectorCustomizer"); - assertThat(customizerDefinition.isLazyInit()).isTrue(); - BeanDefinition connectorDefinition = context.getBeanFactory().getBeanDefinition("webClientHttpConnector"); - assertThat(connectorDefinition.isLazyInit()).isTrue(); - assertThat(context).hasBean("reactorClientHttpConnectorFactory"); - assertThat(context).hasSingleBean(ReactorResourceFactory.class); - }); - } - @Test - void whenReactorIsUnavailableThenHttpClientBeansAreDefined() { - this.contextRunner.withClassLoader(new FilteredClassLoader(HttpClient.class)).run((context) -> { - BeanDefinition customizerDefinition = context.getBeanFactory() - .getBeanDefinition("webClientHttpConnectorCustomizer"); - assertThat(customizerDefinition.isLazyInit()).isTrue(); - BeanDefinition connectorDefinition = context.getBeanFactory().getBeanDefinition("webClientHttpConnector"); - assertThat(connectorDefinition.isLazyInit()).isTrue(); - assertThat(context).hasBean("httpComponentsClientHttpConnectorFactory"); - }); - } - - @Test - void whenReactorAndHttpClientBeansAreUnavailableThenJdkClientBeansAreDefined() { - this.contextRunner.withClassLoader(new FilteredClassLoader(HttpClient.class, HttpAsyncClients.class)) + void shouldApplyReactorNettyHttpClientMapper() { + new ReactiveWebApplicationContextRunner().withConfiguration(AutoConfigurations.of( + ClientHttpConnectorAutoConfiguration.class, + org.springframework.boot.autoconfigure.http.client.reactive.ClientHttpConnectorAutoConfiguration.class)) + .withUserConfiguration(CustomReactorNettyHttpClientMapper.class) .run((context) -> { - BeanDefinition customizerDefinition = context.getBeanFactory() - .getBeanDefinition("webClientHttpConnectorCustomizer"); - assertThat(customizerDefinition.isLazyInit()).isTrue(); - BeanDefinition connectorDefinition = context.getBeanFactory() - .getBeanDefinition("webClientHttpConnector"); - assertThat(connectorDefinition.isLazyInit()).isTrue(); - assertThat(context).hasBean("jdkClientHttpConnectorFactory"); + context.getBean(ClientHttpConnector.class); + assertThat(CustomReactorNettyHttpClientMapper.called).isTrue(); }); } - @Test - void shouldCreateHttpClientBeans() { - this.contextRunner.run((context) -> { - assertThat(context).hasSingleBean(ReactorResourceFactory.class); - assertThat(context).hasSingleBean(ClientHttpConnector.class); - WebClientCustomizer clientCustomizer = context.getBean(WebClientCustomizer.class); - WebClient.Builder builder = mock(WebClient.Builder.class); - clientCustomizer.customize(builder); - then(builder).should().clientConnector(any(ReactorClientHttpConnector.class)); - }); - } - - @Test - void shouldNotOverrideCustomClientConnector() { - this.contextRunner.withUserConfiguration(CustomClientHttpConnectorConfig.class).run((context) -> { - assertThat(context).hasSingleBean(ClientHttpConnector.class).hasBean("customConnector"); - WebClientCustomizer clientCustomizer = context.getBean(WebClientCustomizer.class); - WebClient.Builder builder = mock(WebClient.Builder.class); - clientCustomizer.customize(builder); - then(builder).should().clientConnector(any(ClientHttpConnector.class)); - }); - } - - @Test - void shouldNotOverrideCustomClientConnectorFactory() { - this.contextRunner.withUserConfiguration(CustomClientHttpConnectorFactoryConfig.class).run((context) -> { - assertThat(context).hasSingleBean(ClientHttpConnectorFactory.class) - .hasBean("customConnector") - .doesNotHaveBean(ReactorResourceFactory.class); - WebClientCustomizer clientCustomizer = context.getBean(WebClientCustomizer.class); - WebClient.Builder builder = mock(WebClient.Builder.class); - clientCustomizer.customize(builder); - then(builder).should().clientConnector(any(ClientHttpConnector.class)); - }); - } - - @Test - void shouldUseCustomReactorResourceFactory() { - this.contextRunner.withUserConfiguration(CustomReactorResourceConfig.class) - .run((context) -> assertThat(context).hasSingleBean(ClientHttpConnector.class) - .hasSingleBean(ReactorResourceFactory.class) - .hasBean("customReactorResourceFactory")); - } - - @Configuration(proxyBeanMethods = false) - static class CustomClientHttpConnectorConfig { - - @Bean - ClientHttpConnector customConnector() { - return mock(ClientHttpConnector.class); - } - - } - - @Configuration(proxyBeanMethods = false) - static class CustomClientHttpConnectorFactoryConfig { - - @Bean - ClientHttpConnectorFactory customConnector() { - return (sslBundle) -> mock(ClientHttpConnector.class); - } - - } + static class CustomReactorNettyHttpClientMapper { - @Configuration(proxyBeanMethods = false) - static class CustomReactorResourceConfig { + static boolean called = false; @Bean - ReactorResourceFactory customReactorResourceFactory() { - return new ReactorResourceFactory(); + ReactorNettyHttpClientMapper clientMapper() { + return (client) -> { + called = true; + return client.baseUrl("/test"); + }; } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorFactoryConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorFactoryConfigurationTests.java deleted file mode 100644 index 78b02ef97d9c..000000000000 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ClientHttpConnectorFactoryConfigurationTests.java +++ /dev/null @@ -1,83 +0,0 @@ -/* - * Copyright 2012-2025 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.autoconfigure.web.reactive.function.client; - -import org.junit.jupiter.api.Test; - -import org.springframework.boot.autoconfigure.AutoConfigurations; -import org.springframework.boot.ssl.SslBundle; -import org.springframework.boot.ssl.SslBundleKey; -import org.springframework.boot.ssl.jks.JksSslStoreBundle; -import org.springframework.boot.ssl.jks.JksSslStoreDetails; -import org.springframework.boot.test.context.FilteredClassLoader; -import org.springframework.boot.test.context.runner.ReactiveWebApplicationContextRunner; -import org.springframework.boot.testsupport.classpath.resources.WithPackageResources; -import org.springframework.context.annotation.Bean; -import org.springframework.http.client.reactive.HttpComponentsClientHttpConnector; - -import static org.assertj.core.api.Assertions.assertThat; -import static org.mockito.BDDMockito.then; -import static org.mockito.Mockito.spy; - -/** - * Tests for {@link ClientHttpConnectorFactoryConfiguration}. - * - * @author Phillip Webb - * @author Brian Clozel - * @author Moritz Halbritter - */ -class ClientHttpConnectorFactoryConfigurationTests { - - @Test - @WithPackageResources("test.jks") - void shouldApplyHttpClientMapper() { - JksSslStoreDetails storeDetails = JksSslStoreDetails.forLocation("classpath:test.jks"); - JksSslStoreBundle stores = new JksSslStoreBundle(storeDetails, storeDetails); - SslBundle sslBundle = spy(SslBundle.of(stores, SslBundleKey.of("password"))); - new ReactiveWebApplicationContextRunner() - .withConfiguration(AutoConfigurations.of(ClientHttpConnectorFactoryConfiguration.ReactorNetty.class)) - .withUserConfiguration(CustomHttpClientMapper.class) - .run((context) -> { - context.getBean(ReactorClientHttpConnectorFactory.class).createClientHttpConnector(sslBundle); - assertThat(CustomHttpClientMapper.called).isTrue(); - then(sslBundle).should().getManagers(); - }); - } - - @Test - void shouldNotConfigureReactiveHttpClient5WhenHttpCore5ReactiveJarIsMissing() { - new ReactiveWebApplicationContextRunner() - .withClassLoader(new FilteredClassLoader("org.apache.hc.core5.reactive")) - .withConfiguration(AutoConfigurations.of(ClientHttpConnectorFactoryConfiguration.HttpClient5.class)) - .run((context) -> assertThat(context).doesNotHaveBean(HttpComponentsClientHttpConnector.class)); - } - - static class CustomHttpClientMapper { - - static boolean called = false; - - @Bean - ReactorNettyHttpClientMapper clientMapper() { - return (client) -> { - called = true; - return client.baseUrl("/test"); - }; - } - - } - -} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorClientHttpConnectorFactoryTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorClientHttpConnectorFactoryTests.java deleted file mode 100644 index 951941d02446..000000000000 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorClientHttpConnectorFactoryTests.java +++ /dev/null @@ -1,49 +0,0 @@ -/* - * Copyright 2012-2023 the original author or authors. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * https://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package org.springframework.boot.autoconfigure.web.reactive.function.client; - -import org.junit.jupiter.api.AfterEach; -import org.junit.jupiter.api.BeforeEach; - -import org.springframework.http.client.ReactorResourceFactory; - -/** - * Tests for {@link ReactorClientHttpConnectorFactory}. - * - * @author Phillip Webb - */ -class ReactorClientHttpConnectorFactoryTests extends AbstractClientHttpConnectorFactoryTests { - - private ReactorResourceFactory resourceFactory; - - @BeforeEach - void setup() { - this.resourceFactory = new ReactorResourceFactory(); - this.resourceFactory.afterPropertiesSet(); - } - - @AfterEach - void teardown() { - this.resourceFactory.destroy(); - } - - @Override - protected ClientHttpConnectorFactory getFactory() { - return new ReactorClientHttpConnectorFactory(this.resourceFactory); - } - -} diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorNettyHttpClientMapperTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorNettyHttpClientMapperTests.java index a49c95a6f4a0..81c84bcc2ea5 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorNettyHttpClientMapperTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/ReactorNettyHttpClientMapperTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,6 +31,7 @@ * * @author Phillip Webb */ +@SuppressWarnings("removal") class ReactorNettyHttpClientMapperTests { @Test @@ -47,7 +48,7 @@ void ofWithCollectionCreatesComposite() { void ofWhenCollectionIsNullThrowsException() { Collection mappers = null; assertThatIllegalArgumentException().isThrownBy(() -> ReactorNettyHttpClientMapper.of(mappers)) - .withMessage("Mappers must not be null"); + .withMessage("'mappers' must not be null"); } @Test @@ -64,7 +65,7 @@ void ofWithArrayCreatesComposite() { void ofWhenArrayIsNullThrowsException() { ReactorNettyHttpClientMapper[] mappers = null; assertThatIllegalArgumentException().isThrownBy(() -> ReactorNettyHttpClientMapper.of(mappers)) - .withMessage("Mappers must not be null"); + .withMessage("'mappers' must not be null"); } private static class TestHttpClient extends HttpClient { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/WebClientAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/WebClientAutoConfigurationTests.java index 853b1fb6ef82..371565348bfb 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/WebClientAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/function/client/WebClientAutoConfigurationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,8 @@ class WebClientAutoConfigurationTests { private final ApplicationContextRunner contextRunner = new ApplicationContextRunner() - .withConfiguration(AutoConfigurations.of(ClientHttpConnectorAutoConfiguration.class, + .withConfiguration(AutoConfigurations.of( + org.springframework.boot.autoconfigure.http.client.reactive.ClientHttpConnectorAutoConfiguration.class, WebClientAutoConfiguration.class, SslAutoConfiguration.class)); @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/DispatcherServletRegistrationBeanTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/DispatcherServletRegistrationBeanTests.java index 6fef92dd72f5..8eec73005f87 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/DispatcherServletRegistrationBeanTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/DispatcherServletRegistrationBeanTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ class DispatcherServletRegistrationBeanTests { void createWhenPathIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new DispatcherServletRegistrationBean(new DispatcherServlet(), null)) - .withMessageContaining("Path must not be null"); + .withMessageContaining("'path' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcAutoConfigurationTests.java index 5a3acfa30443..a2def5a17b11 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcAutoConfigurationTests.java @@ -81,6 +81,7 @@ import org.springframework.format.support.FormattingConversionService; import org.springframework.http.CacheControl; import org.springframework.http.HttpHeaders; +import org.springframework.http.MediaType; import org.springframework.http.converter.HttpMessageConverter; import org.springframework.mock.web.MockHttpServletRequest; import org.springframework.test.util.ReflectionTestUtils; @@ -88,6 +89,7 @@ import org.springframework.validation.Validator; import org.springframework.validation.beanvalidation.LocalValidatorFactoryBean; import org.springframework.web.accept.ContentNegotiationManager; +import org.springframework.web.accept.FixedContentNegotiationStrategy; import org.springframework.web.accept.ParameterContentNegotiationStrategy; import org.springframework.web.bind.annotation.ControllerAdvice; import org.springframework.web.bind.support.ConfigurableWebBindingInitializer; @@ -571,12 +573,23 @@ void asyncTaskExecutorWithCustomNonApplicationTaskExecutor() { @Test void customMediaTypes() { this.contextRunner.withPropertyValues("spring.mvc.contentnegotiation.media-types.yaml:text/yaml") - .run((context) -> { - RequestMappingHandlerAdapter adapter = context.getBean(RequestMappingHandlerAdapter.class); - ContentNegotiationManager contentNegotiationManager = (ContentNegotiationManager) ReflectionTestUtils - .getField(adapter, "contentNegotiationManager"); - assertThat(contentNegotiationManager.getAllFileExtensions()).contains("yaml"); - }); + .run((context) -> assertThat(context.getBean(RequestMappingHandlerAdapter.class)) + .extracting("contentNegotiationManager", + InstanceOfAssertFactories.type(ContentNegotiationManager.class)) + .satisfies((contentNegotiationManager) -> assertThat(contentNegotiationManager.getAllFileExtensions()) + .contains("yaml"))); + } + + @Test + void customDefaultContentTypes() { + this.contextRunner + .withPropertyValues("spring.mvc.contentnegotiation.default-content-types:application/json,application/xml") + .run((context) -> assertThat(context.getBean(RequestMappingHandlerAdapter.class)) + .extracting("contentNegotiationManager", + InstanceOfAssertFactories.type(ContentNegotiationManager.class)) + .satisfies((contentNegotiationManager) -> assertThat( + contentNegotiationManager.getStrategy(FixedContentNegotiationStrategy.class).getContentTypes()) + .containsExactly(MediaType.APPLICATION_JSON, MediaType.APPLICATION_XML))); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcPropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcPropertiesTests.java index 5b7a014bd115..c4d129ce7be1 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcPropertiesTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -58,7 +58,7 @@ void servletPathWhenDoesNotEndWithSlashHasValidMappingAndPrefix() { void servletPathWhenHasWildcardThrowsException() { assertThatExceptionOfType(BindException.class).isThrownBy(() -> bind("spring.mvc.servlet.path", "/*")) .withRootCauseInstanceOf(IllegalArgumentException.class) - .satisfies((ex) -> assertThat(Throwables.getRootCause(ex)).hasMessage("Path must not contain wildcards")); + .satisfies((ex) -> assertThat(Throwables.getRootCause(ex)).hasMessage("'path' must not contain wildcards")); } private void bind(String name, String value) { diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/error/DefaultErrorViewResolverTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/error/DefaultErrorViewResolverTests.java index 1f4b6e4599d7..beff55782c08 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/error/DefaultErrorViewResolverTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/servlet/error/DefaultErrorViewResolverTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -83,14 +83,14 @@ void setup() { @Test void createWhenApplicationContextIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new DefaultErrorViewResolver(null, new Resources())) - .withMessageContaining("ApplicationContext must not be null"); + .withMessageContaining("'applicationContext' must not be null"); } @Test void createWhenResourcePropertiesIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new DefaultErrorViewResolver(mock(ApplicationContext.class), (Resources) null)) - .withMessageContaining("Resources must not be null"); + .withMessageContaining("'resources' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/webservices/WebServicesAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/webservices/WebServicesAutoConfigurationTests.java index d00d451819ab..486e1da2b4e7 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/webservices/WebServicesAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/webservices/WebServicesAutoConfigurationTests.java @@ -57,7 +57,7 @@ void customPathMustBeginWithASlash() { .run((context) -> assertThat(context).getFailure() .isInstanceOf(BeanCreationException.class) .rootCause() - .hasMessageContaining("Path must start with '/'")); + .hasMessageContaining("'path' must start with '/'")); } @Test diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/webservices/WebServicesPropertiesTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/webservices/WebServicesPropertiesTests.java index c1450a1f46c3..4b94b53c729e 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/webservices/WebServicesPropertiesTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/webservices/WebServicesPropertiesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,21 +33,21 @@ class WebServicesPropertiesTests { void pathMustNotBeEmpty() { this.properties = new WebServicesProperties(); assertThatIllegalArgumentException().isThrownBy(() -> this.properties.setPath("")) - .withMessageContaining("Path must have length greater than 1"); + .withMessageContaining("'path' must have length greater than 1"); } @Test void pathMustHaveLengthGreaterThanOne() { this.properties = new WebServicesProperties(); assertThatIllegalArgumentException().isThrownBy(() -> this.properties.setPath("/")) - .withMessageContaining("Path must have length greater than 1"); + .withMessageContaining("'path' must have length greater than 1"); } @Test void customPathMustBeginWithASlash() { this.properties = new WebServicesProperties(); assertThatIllegalArgumentException().isThrownBy(() -> this.properties.setPath("custom")) - .withMessageContaining("Path must start with '/'"); + .withMessageContaining("'path' must start with '/'"); } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/resources/org/springframework/boot/autoconfigure/cache/hazelcast-specific.xml b/spring-boot-project/spring-boot-autoconfigure/src/test/resources/org/springframework/boot/autoconfigure/cache/hazelcast-specific.xml new file mode 100644 index 000000000000..f5a30301dca4 --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/resources/org/springframework/boot/autoconfigure/cache/hazelcast-specific.xml @@ -0,0 +1,19 @@ + + + + + + 3600 + 600 + + + + + + + + + + diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/resources/org/springframework/boot/autoconfigure/jooq/settings.xml b/spring-boot-project/spring-boot-autoconfigure/src/test/resources/org/springframework/boot/autoconfigure/jooq/settings.xml new file mode 100644 index 000000000000..ee57678ae40b --- /dev/null +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/resources/org/springframework/boot/autoconfigure/jooq/settings.xml @@ -0,0 +1,4 @@ + + + 100 + diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/resources/org/springframework/boot/autoconfigure/web/reactive/function/client/test.jks b/spring-boot-project/spring-boot-autoconfigure/src/test/resources/org/springframework/boot/autoconfigure/web/reactive/function/client/test.jks deleted file mode 100644 index 0fc3e802f754..000000000000 Binary files a/spring-boot-project/spring-boot-autoconfigure/src/test/resources/org/springframework/boot/autoconfigure/web/reactive/function/client/test.jks and /dev/null differ diff --git a/spring-boot-project/spring-boot-dependencies/build.gradle b/spring-boot-project/spring-boot-dependencies/build.gradle index 4e81187d0a97..c214175541fc 100644 --- a/spring-boot-project/spring-boot-dependencies/build.gradle +++ b/spring-boot-project/spring-boot-dependencies/build.gradle @@ -6,9 +6,8 @@ plugins { description = "Spring Boot Dependencies" bom { - effectiveBomArtifact() upgrade { - policy = "same-minor-version" + policy = "any" gitHub { issueLabels = ["type: dependency-upgrade"] } @@ -23,9 +22,7 @@ bom { exclude group: "commons-logging", module: "commons-logging" } ] - imports = [ - "activemq-bom" - ] + bom("activemq-bom") } links { site("https://activemq.apache.org") @@ -53,11 +50,11 @@ bom { releaseNotes("https://github.com/eclipse-ee4j/angus-mail/releases/tag/{version}") } } - library("Artemis", "2.37.0") { + library("Artemis", "2.40.0") { group("org.apache.activemq") { - imports = [ - "artemis-bom" - ] + bom("artemis-bom") { + permit("org.apache.maven.plugin-tools:maven-plugin-annotations") + } } links { site("https://activemq.apache.org/components/artemis") @@ -81,16 +78,14 @@ bom { } library("AssertJ", "${assertjVersion}") { group("org.assertj") { - imports = [ - "assertj-bom" - ] + bom("assertj-bom") } links { site("https://assertj.github.io/doc") releaseNotes("https://github.com/assertj/assertj/releases/tag/assertj-build-{version}") } } - library("Awaitility", "4.2.2") { + library("Awaitility", "4.3.0") { group("org.awaitility") { modules = [ "awaitility", @@ -104,22 +99,18 @@ bom { .formatted(version.major(), version.minor())) } } - library("Zipkin Reporter", "3.4.3") { + library("Zipkin Reporter", "3.5.0") { group("io.zipkin.reporter2") { - imports = [ - "zipkin-reporter-bom" - ] + bom("zipkin-reporter-bom") } links { site("https://github.com/openzipkin/zipkin-reporter-java") releaseNotes("https://github.com/openzipkin/zipkin-reporter-java/releases/tag/{version}") } } - library("Brave", "6.0.3") { + library("Brave", "6.1.0") { group("io.zipkin.brave") { - imports = [ - "brave-bom" - ] + bom("brave-bom") } links { site("https://github.com/openzipkin/brave") @@ -137,7 +128,7 @@ bom { releaseNotes("https://github.com/mojohaus/build-helper-maven-plugin/releases/tag/{version}") } } - library("Byte Buddy", "1.15.11") { + library("Byte Buddy", "1.17.2") { group("net.bytebuddy") { modules = [ "byte-buddy", @@ -166,7 +157,7 @@ bom { releaseNotes("https://github.com/cache2k/cache2k/releases/tag/v{version}") } } - library("Caffeine", "3.1.8") { + library("Caffeine", "3.2.0") { group("com.github.ben-manes.caffeine") { modules = [ "caffeine", @@ -182,11 +173,11 @@ bom { releaseNotes("https://github.com/ben-manes/caffeine/releases/tag/v{version}") } } - library("Cassandra Driver", "4.18.1") { + library("Cassandra Driver", "4.19.0") { group("org.apache.cassandra") { - imports = [ - "java-driver-bom" - ] + bom("java-driver-bom") { + permit("com.datastax.oss:native-protocol") + } modules = [ "java-driver-core" ] @@ -213,7 +204,7 @@ bom { releaseNotes("https://commons.apache.org/proper/commons-codec/changes-report.html#a{version}") } } - library("Commons DBCP2", "2.12.0") { + library("Commons DBCP2", "2.13.0") { group("org.apache.commons") { modules = [ "commons-dbcp2" { @@ -283,7 +274,7 @@ bom { releaseNotes("https://github.com/CycloneDX/cyclonedx-maven-plugin/releases/tag/cyclonedx-maven-plugin-{version}") } } - library("DB2 JDBC", "11.5.9.0") { + library("DB2 JDBC", "12.1.0.0") { group("com.ibm.db2") { modules = [ "jcc" @@ -337,7 +328,13 @@ bom { releaseNotes("https://github.com/ehcache/ehcache3/releases/tag/v{version}") } } - library("Elasticsearch Client", "8.15.5") { + library("Elasticsearch Client", "8.17.2") { + alignWith { + version { + from "org.springframework.data:spring-data-elasticsearch" + managedBy "Spring Data Bom" + } + } group("org.elasticsearch.client") { modules = [ "elasticsearch-rest-client" { @@ -360,7 +357,7 @@ bom { javadoc("elasticsearch-rest-client-sniffer", version -> "https://artifacts.elastic.co/javadoc/org/elasticsearch/client/elasticsearch-rest-client-sniffer/%s".formatted(version), "org.elasticsearch.client.sniff") } } - library("Flyway", "10.20.1") { + library("Flyway", "11.4.0") { group("org.flywaydb") { modules = [ "flyway-commandline", @@ -419,9 +416,19 @@ bom { } library("Glassfish JAXB", "4.0.5") { group("org.glassfish.jaxb") { - imports = [ - "jaxb-bom" - ] + bom("jaxb-bom") { + permit("com.sun.istack:istack-commons-runtime") + permit("com.sun.xml.bind:jaxb-core") + permit("com.sun.xml.bind:jaxb-impl") + permit("com.sun.xml.bind:jaxb-jxc") + permit("com.sun.xml.bind:jaxb-osgi") + permit("com.sun.xml.bind:jaxb-xjc") + permit("com.sun.xml.fastinfoset:FastInfoset") + permit("jakarta.activation:jakarta.activation-api") + permit("jakarta.xml.bind:jakarta.xml.bind-api") + permit("org.eclipse.angus:angus-activation") + permit("org.jvnet.staxex:stax-ex") + } } links { releaseNotes("https://github.com/eclipse-ee4j/jaxb-ri/releases/tag/{version}-RI") @@ -456,16 +463,18 @@ bom { } } library("Groovy", "4.0.26") { + prohibit { + contains "-alpha-" + because "we don't want alpha dependencies" + } group("org.apache.groovy") { - imports = [ - "groovy-bom" - ] + bom("groovy-bom") } links { site("https://groovy-lang.org") } } - library("Gson", "2.11.0") { + library("Gson", "2.12.1") { group("com.google.code.gson") { modules = [ "gson" @@ -515,6 +524,10 @@ bom { } } library("Hibernate", "6.6.11.Final") { + prohibit { + versionRange "[7.0.0.Alpha1,)" + because "it exceeds our Jakarta EE 10 baseline" + } group("org.hibernate.orm") { modules = [ "hibernate-agroal", @@ -547,6 +560,10 @@ bom { } } library("Hibernate Validator", "8.0.2.Final") { + prohibit { + versionRange "[9.0.0.Beta2,)" + because "it exceeds our Jakarta EE 10 baseline" + } group("org.hibernate.validator") { modules = [ "hibernate-validator", @@ -554,7 +571,7 @@ bom { ] } } - library("HikariCP", "5.1.0") { + library("HikariCP", "6.2.1") { group("com.zaxxer") { modules = [ "HikariCP" @@ -576,7 +593,7 @@ bom { ] } } - library("HtmlUnit", "4.5.0") { + library("HtmlUnit", "4.10.0") { group("org.htmlunit") { modules = [ "htmlunit" { @@ -624,11 +641,9 @@ bom { ] } } - library("Infinispan", "15.0.14.Final") { + library("Infinispan", "15.1.7.Final") { group("org.infinispan") { - imports = [ - "infinispan-bom" - ] + bom("infinispan-bom") } links { site("https://infinispan.org") @@ -650,9 +665,7 @@ bom { } library("Jackson Bom", "${jacksonVersion}") { group("com.fasterxml.jackson") { - imports = [ - "jackson-bom" - ] + bom("jackson-bom") } links { releaseNotes("https://github.com/FasterXML/jackson/wiki/Jackson-Release-{version}") @@ -672,6 +685,10 @@ bom { } } library("Jakarta Annotation", "2.1.1") { + prohibit { + versionRange "[3.0.0-M1,)" + because "it exceeds our Jakarta EE 10 baseline" + } group("jakarta.annotation") { modules = [ "jakarta.annotation-api" @@ -752,8 +769,8 @@ bom { } library("Jakarta Persistence", "3.1.0") { prohibit { - versionRange "[3.2.0-B01,3.2.0]" - because "it's part of Jakarta EE 11" + versionRange "[3.2.0-B01,)" + because "it exceeds our Jakarta EE 10 baseline" } group("jakarta.persistence") { modules = [ @@ -771,8 +788,8 @@ bom { } library("Jakarta Servlet", "6.0.0") { prohibit { - versionRange "[6.1.0-M1,6.1.0]" - because "it's part of Jakarta EE 11" + versionRange "[6.1.0-M1,)" + because "it exceeds our Jakarta EE 10 baseline" } group("jakarta.servlet") { modules = [ @@ -809,8 +826,8 @@ bom { } library("Jakarta Validation", "3.0.2") { prohibit { - versionRange "[3.1.0-M1,3.1.0]" - because "it's part of Jakarta EE 11" + versionRange "[3.1.0-M1,)" + because "it exceeds our Jakarta EE 10 baseline" } group("jakarta.validation") { modules = [ @@ -825,8 +842,8 @@ bom { } library("Jakarta WebSocket", "2.1.1") { prohibit { - versionRange "[2.2.0-M1,2.2.0]" - because "it's part of Jakarta EE 11" + versionRange "[2.2.0-M1,)" + because "it exceeds our Jakarta EE 10 baseline" } group("jakarta.websocket") { modules = [ @@ -843,6 +860,10 @@ bom { } } library("Jakarta WS RS", "3.1.0") { + prohibit { + versionRange "[4.0.0-M2,)" + because "it exceeds our Jakarta EE 10 baseline" + } group("jakarta.ws.rs") { modules = [ "jakarta.ws.rs-api" @@ -922,7 +943,7 @@ bom { releaseNotes("https://github.com/jaxen-xpath/jaxen/releases/tag/v{version}") } } - library("Jaybird", "5.0.6.java11") { + library("Jaybird", "6.0.0") { prohibit { endsWith ".java8" because "we use the .java11 version" @@ -958,6 +979,10 @@ bom { } } library("Jedis", "5.2.0") { + prohibit { + contains "-beta" + because "we don't want beta dependencies" + } group("redis.clients") { modules = [ "jedis" @@ -969,10 +994,12 @@ bom { } } library("Jersey", "3.1.10") { + prohibit { + versionRange "[4.0.0-M1,)" + because "it exceeds our Jakarta EE 10 baseline" + } group("org.glassfish.jersey") { - imports = [ - "jersey-bom" - ] + bom("jersey-bom") } links { site("https://github.com/eclipse-ee4j/jersey") @@ -988,15 +1015,15 @@ bom { } } library("Jetty", "12.0.18") { + prohibit { + contains ".alpha" + because "we don't want alpha dependencies" + } group("org.eclipse.jetty.ee10") { - imports = [ - "jetty-ee10-bom" - ] + bom("jetty-ee10-bom") } group("org.eclipse.jetty") { - imports = [ - "jetty-bom" - ] + bom("jetty-bom") } links { site("https://eclipse.dev/jetty") @@ -1012,6 +1039,10 @@ bom { } } library("jOOQ", "3.19.21") { + prohibit { + versionRange "[3.20.0,)" + because "it requires Java 21" + } group("org.jooq") { modules = [ "jooq", @@ -1054,6 +1085,10 @@ bom { } } library("JsonAssert", "1.5.3") { + prohibit { + contains "-rc" + because "we don't want release candidates" + } group("org.skyscreamer") { modules = [ "jsonassert" @@ -1082,10 +1117,12 @@ bom { } } library("JUnit Jupiter", "${junitJupiterVersion}") { + prohibit { + contains "-M" + because "we don't want milestones" + } group("org.junit") { - imports = [ - "junit-bom" - ] + bom("junit-bom") } links { site("https://junit.org/junit5") @@ -1095,7 +1132,7 @@ bom { releaseNotes("https://junit.org/junit5/docs/{version}/release-notes") } } - library("Kafka", "3.8.1") { + library("Kafka", "3.9.0") { group("org.apache.kafka") { modules = [ "connect", @@ -1146,10 +1183,12 @@ bom { } } library("Kotlin", "${kotlinVersion}") { + prohibit { + versionRange "[2.0.0-Beta1,)" + because "it exceeds our baseline" + } group("org.jetbrains.kotlin") { - imports = [ - "kotlin-bom" - ] + bom("kotlin-bom") plugins = [ "kotlin-maven-plugin" ] @@ -1162,13 +1201,11 @@ bom { } library("Kotlin Coroutines", "1.8.1") { prohibit { - versionRange "[1.9.0,)" + versionRange "[1.9.0-RC,)" because "it requires Kotlin 2" } group("org.jetbrains.kotlinx") { - imports = [ - "kotlinx-coroutines-bom" - ] + bom("kotlinx-coroutines-bom") } links { site("https://github.com/Kotlin/kotlinx.coroutines") @@ -1181,16 +1218,18 @@ bom { because "it requires Kotlin 2" } group("org.jetbrains.kotlinx") { - imports = [ - "kotlinx-serialization-bom" - ] + bom("kotlinx-serialization-bom") } links { site("https://github.com/Kotlin/kotlinx.serialization") releaseNotes("https://github.com/Kotlin/kotlinx.serialization/releases/tag/v{version}") } } - library("Lettuce", "6.4.2.RELEASE") { + library("Lettuce", "6.5.5.RELEASE") { + prohibit { + contains ".BETA" + because "we don't want betas" + } group("io.lettuce") { modules = [ "lettuce-core" @@ -1203,7 +1242,7 @@ bom { releaseNotes("https://github.com/lettuce-io/lettuce-core/releases/tag/{version}") } } - library("Liquibase", "4.29.2") { + library("Liquibase", "4.31.1") { group("org.liquibase") { modules = [ "liquibase-cdi", @@ -1220,10 +1259,22 @@ bom { } } library("Log4j2", "2.24.3") { + prohibit { + contains "-alpha" + contains "-beta" + because "we don't want alphas or betas" + } group("org.apache.logging.log4j") { - imports = [ - "log4j-bom" - ] + bom("log4j-bom") { + permit("biz.aQute.bnd:biz.aQute.bnd.annotation") + permit("com.github.spotbugs:spotbugs-annotations") + permit("org.apache.logging:logging-parent") + permit("org.apache.maven.plugin-tools:maven-plugin-annotations") + permit("org.jspecify:jspecify") + permit("org.osgi:org.osgi.annotation.bundle") + permit("org.osgi:org.osgi.annotation.versioning") + permit("org.osgi:osgi.annotation") + } } links { site("https://logging.apache.org/log4j") @@ -1256,7 +1307,7 @@ bom { javadoc("https://projectlombok.org/api") } } - library("MariaDB", "3.4.1") { + library("MariaDB", "3.5.2") { group("org.mariadb.jdbc") { modules = [ "mariadb-java-client" @@ -1286,6 +1337,10 @@ bom { } } library("Maven Clean Plugin", "3.4.1") { + prohibit { + contains "-beta-" + because "we don't want betas" + } group("org.apache.maven.plugins") { plugins = [ "maven-clean-plugin" @@ -1295,7 +1350,11 @@ bom { releaseNotes("https://github.com/apache/maven-clean-plugin/releases/tag/maven-clean-plugin-{version}") } } - library("Maven Compiler Plugin", "3.13.0") { + library("Maven Compiler Plugin", "3.14.0") { + prohibit { + contains "-beta-" + because "we don't want betas" + } group("org.apache.maven.plugins") { plugins = [ "maven-compiler-plugin" @@ -1316,6 +1375,10 @@ bom { } } library("Maven Deploy Plugin", "3.1.4") { + prohibit { + contains "-beta-" + because "we don't want betas" + } group("org.apache.maven.plugins") { plugins = [ "maven-deploy-plugin" @@ -1353,6 +1416,10 @@ bom { } } library("Maven Install Plugin", "3.1.4") { + prohibit { + contains "-beta-" + because "we don't want betas" + } group("org.apache.maven.plugins") { plugins = [ "maven-install-plugin" @@ -1362,7 +1429,7 @@ bom { releaseNotes("https://github.com/apache/maven-install-plugin/releases/tag/maven-install-plugin-{version}") } } - library("Maven Invoker Plugin", "3.8.1") { + library("Maven Invoker Plugin", "3.9.0") { group("org.apache.maven.plugins") { plugins = [ "maven-invoker-plugin" @@ -1373,6 +1440,10 @@ bom { } } library("Maven Jar Plugin", "3.4.2") { + prohibit { + contains "-beta-" + because "we don't want betas" + } group("org.apache.maven.plugins") { plugins = [ "maven-jar-plugin" @@ -1382,7 +1453,7 @@ bom { releaseNotes("https://github.com/apache/maven-jar-plugin/releases/tag/maven-jar-plugin-{version}") } } - library("Maven Javadoc Plugin", "3.10.1") { + library("Maven Javadoc Plugin", "3.11.2") { group("org.apache.maven.plugins") { plugins = [ "maven-javadoc-plugin" @@ -1393,6 +1464,10 @@ bom { } } library("Maven Resources Plugin", "3.3.1") { + prohibit { + contains "-beta-" + because "we don't want betas" + } group("org.apache.maven.plugins") { plugins = [ "maven-resources-plugin" @@ -1413,6 +1488,10 @@ bom { } } library("Maven Source Plugin", "3.3.1") { + prohibit { + contains "-beta-" + because "we don't want betas" + } group("org.apache.maven.plugins") { plugins = [ "maven-source-plugin" @@ -1442,7 +1521,7 @@ bom { releaseNotes("https://github.com/apache/maven-war-plugin/releases/tag/maven-war-plugin-{version}") } } - library("Micrometer", "1.14.5") { + library("Micrometer", "1.15.0-M3") { considerSnapshots() group("io.micrometer") { modules = [ @@ -1450,9 +1529,7 @@ bom { exclude group: "javax.annotation", module: "javax.annotation-api" } ] - imports = [ - "micrometer-bom" - ] + bom("micrometer-bom") } links { site("https://micrometer.io") @@ -1466,12 +1543,10 @@ bom { releaseNotes("https://github.com/micrometer-metrics/micrometer/releases/tag/v{version}") } } - library("Micrometer Tracing", "1.4.4") { + library("Micrometer Tracing", "1.5.0-M3") { considerSnapshots() group("io.micrometer") { - imports = [ - "micrometer-tracing-bom" - ] + bom("micrometer-tracing-bom") } links { site("https://micrometer.io") @@ -1483,16 +1558,14 @@ bom { } library("Mockito", "${mockitoVersion}") { group("org.mockito") { - imports = [ - "mockito-bom" - ] + bom("mockito-bom") } links { site("https://site.mockito.org") releaseNotes("https://github.com/mockito/mockito/releases/tag/v{version}") } } - library("MongoDB", "5.2.1") { + library("MongoDB", "5.3.1") { group("org.mongodb") { modules = [ "bson", @@ -1511,7 +1584,7 @@ bom { releaseNotes("https://github.com/mongodb/mongo-java-driver/releases/tag/r{version}") } } - library("MSSQL JDBC", "12.8.1.jre11") { + library("MSSQL JDBC", "12.10.0.jre11") { prohibit { endsWith(".jre8") because "we want to use the jre11 version" @@ -1531,7 +1604,7 @@ bom { .formatted(version.toString().replace(".jre11", ""))) } } - library("MySQL", "9.1.0") { + library("MySQL", "9.2.0") { group("com.mysql") { modules = [ "mysql-connector-j" { @@ -1581,21 +1654,23 @@ bom { } } library("Netty", "4.1.119.Final") { + prohibit { + contains ".Alpha" + contains ".Beta" + contains ".RC" + because "we don't want alphas, betas, or release candidates" + } group("io.netty") { - imports = [ - "netty-bom" - ] + bom("netty-bom") } links { site("https://netty.io") javadoc(version -> "https://netty.io/%s.%s/api".formatted(version.major(), version.minor()), "io.netty") } } - library("OpenTelemetry", "1.43.0") { + library("OpenTelemetry", "1.48.0") { group("io.opentelemetry") { - imports = [ - "opentelemetry-bom" - ] + bom("opentelemetry-bom") } links { site("https://github.com/open-telemetry/opentelemetry-java") @@ -1608,7 +1683,7 @@ bom { releaseNotes("https://github.com/open-telemetry/opentelemetry-java/releases/tag/v{version}") } } - library("Oracle Database", "23.5.0.24.07") { + library("Oracle Database", "23.7.0.25.01") { alignWith { dependencyManagementDeclaredIn("com.oracle.database.jdbc:ojdbc-bom") } @@ -1622,11 +1697,14 @@ bom { modules = [ "ojdbc11", "ojdbc11-production", + "ojdbc17", + "ojdbc17-production", "ojdbc8", "ojdbc8-production", "rsi", "ucp", - "ucp11" + "ucp11", + "ucp17" ] } group("com.oracle.database.nls") { @@ -1646,7 +1724,7 @@ bom { ] } } - library("Oracle R2DBC", "1.2.0") { + library("Oracle R2DBC", "1.3.0") { group("com.oracle.database.r2dbc") { modules = [ "oracle-r2dbc" @@ -1680,9 +1758,7 @@ bom { } library("Prometheus Client", "1.3.6") { group("io.prometheus") { - imports = [ - "prometheus-metrics-bom" - ] + bom("prometheus-metrics-bom") } links { site("https://github.com/prometheus/client_java") @@ -1692,9 +1768,7 @@ bom { } library("Prometheus Simpleclient", "0.16.0") { group("io.prometheus") { - imports = [ - "simpleclient_bom" - ] + bom("simpleclient_bom") } links { site("https://github.com/prometheus/client_java") @@ -1702,11 +1776,11 @@ bom { releaseNotes("https://github.com/prometheus/client_java/releases/tag/parent-{version}") } } - library("Pulsar", "3.3.5") { + library("Pulsar", "4.0.3") { group("org.apache.pulsar") { - imports = [ - "pulsar-bom" - ] + bom("pulsar-bom") { + permit("org.apache.maven.plugin-tools:maven-plugin-annotations") + } } links { site("https://pulsar.apache.org") @@ -1717,26 +1791,17 @@ bom { } library("Pulsar Reactive", "0.5.10") { group("org.apache.pulsar") { - modules = [ - "pulsar-client-reactive-adapter", - "pulsar-client-reactive-api", - "pulsar-client-reactive-jackson", - "pulsar-client-reactive-producer-cache-caffeine-shaded", - "pulsar-client-reactive-producer-cache-caffeine" - ] + bom("pulsar-client-reactive-bom") } links { site("https://github.com/apache/pulsar-client-reactive") releaseNotes("https://github.com/apache/pulsar-client-reactive/releases/tag/v{version}") } } - library("Quartz", "2.3.2") { + library("Quartz", "2.5.0") { group("org.quartz-scheduler") { modules = [ - "quartz" { - exclude group: "com.mchange", module: "c3p0" - exclude group: "com.zaxxer", module: "*" - }, + "quartz", "quartz-jobs" ] } @@ -1748,9 +1813,7 @@ bom { } library("QueryDSL", "5.1.0") { group("com.querydsl") { - imports = [ - "querydsl-bom" - ] + bom("querydsl-bom") } links { site("https://github.com/querydsl/querydsl") @@ -1769,7 +1832,7 @@ bom { releaseNotes("https://github.com/r2dbc/r2dbc-h2/releases/tag/v{version}") } } - library("R2DBC MariaDB", "1.2.2") { + library("R2DBC MariaDB", "1.3.0") { group("org.mariadb") { modules = [ "r2dbc-mariadb" @@ -1789,7 +1852,7 @@ bom { releaseNotes("https://github.com/r2dbc/r2dbc-mssql/releases/tag/v{version}") } } - library("R2DBC MySQL", "1.3.2") { + library("R2DBC MySQL", "1.4.0") { group("io.asyncer") { modules = [ "r2dbc-mysql" @@ -1846,7 +1909,7 @@ bom { releaseNotes("https://github.com/r2dbc/r2dbc-spi/releases/tag/v{version}") } } - library("Rabbit AMQP Client", "5.22.0") { + library("Rabbit AMQP Client", "5.25.0") { group("com.rabbitmq") { modules = [ "amqp-client" @@ -1858,7 +1921,7 @@ bom { releaseNotes("https://github.com/rabbitmq/rabbitmq-java-client/releases/tag/v{version}") } } - library("Rabbit Stream Client", "0.18.0") { + library("Rabbit Stream Client", "0.22.0") { group("com.rabbitmq") { modules = [ "stream-client" @@ -1880,9 +1943,9 @@ bom { considerSnapshots() calendarName = "Reactor" group("io.projectreactor") { - imports = [ - "reactor-bom" - ] + bom("reactor-bom") { + permit("org.reactivestreams:reactive-streams") + } } links { site("https://projectreactor.io") @@ -1891,9 +1954,7 @@ bom { } library("REST Assured", "5.5.1") { group("io.rest-assured") { - imports = [ - "rest-assured-bom" - ] + bom("rest-assured-bom") } links { javadoc("https://javadoc.io/doc/io.rest-assured/rest-assured/{version}", "io.restassured") @@ -1901,9 +1962,7 @@ bom { } library("RSocket", "1.1.5") { group("io.rsocket") { - imports = [ - "rsocket-bom" - ] + bom("rsocket-bom") } links { site("https://github.com/rsocket/rsocket-java") @@ -2019,11 +2078,9 @@ bom { ] } } - library("Selenium", "4.25.0") { + library("Selenium", "4.29.0") { group("org.seleniumhq.selenium") { - imports = [ - "selenium-bom" - ] + bom("selenium-bom") } links { site("https://www.selenium.dev") @@ -2031,7 +2088,7 @@ bom { releaseNotes("https://github.com/SeleniumHQ/selenium/releases/tag/selenium-{version}") } } - library("Selenium HtmlUnit", "4.25.0") { + library("Selenium HtmlUnit", "4.29.0") { group("org.seleniumhq.selenium") { modules = [ "htmlunit3-driver" @@ -2043,6 +2100,10 @@ bom { } } library("SendGrid", "4.10.3") { + prohibit { + contains "-rc." + because "we don't want release candidates" + } group("com.sendgrid") { modules = [ "sendgrid-java" @@ -2054,6 +2115,10 @@ bom { } } library("SLF4J", "2.0.17") { + prohibit { + contains "-alpha" + because "we don't want alphas" + } group("org.slf4j") { modules = [ "jcl-over-slf4j", @@ -2080,9 +2145,7 @@ bom { library("Spring AMQP", "3.2.4") { considerSnapshots() group("org.springframework.amqp") { - imports = [ - "spring-amqp-bom" - ] + bom("spring-amqp-bom") } links { site("https://spring.io/projects/spring-amqp") @@ -2094,7 +2157,7 @@ bom { releaseNotes("https://github.com/spring-projects/spring-amqp/releases/tag/v{version}") } } - library("Spring Authorization Server", "1.4.2") { + library("Spring Authorization Server", "1.5.0-M2") { considerSnapshots() group("org.springframework.security") { modules = [ @@ -2114,9 +2177,7 @@ bom { library("Spring Batch", "5.2.2") { considerSnapshots() group("org.springframework.batch") { - imports = [ - "spring-batch-bom" - ] + bom("spring-batch-bom") } links { site("https://spring.io/projects/spring-batch") @@ -2128,13 +2189,15 @@ bom { releaseNotes("https://github.com/spring-projects/spring-batch/releases/tag/v{version}") } } - library("Spring Data Bom", "2024.1.4") { + library("Spring Data Bom", "2025.0.0-M2") { + prohibit { + versionRange "[2025.1.0-M1,)" + because "it exceeds our baseline" + } considerSnapshots() calendarName = "Spring Data Release" group("org.springframework.data") { - imports = [ - "spring-data-bom" - ] + bom("spring-data-bom") } links("spring-data") { site("https://spring.io/projects/spring-data") @@ -2143,11 +2206,13 @@ bom { } } library("Spring Framework", "${springFrameworkVersion}") { + prohibit { + versionRange "[7.0.0-M1,)" + because "it exceeds our baseline" + } considerSnapshots() group("org.springframework") { - imports = [ - "spring-framework-bom" - ] + bom("spring-framework-bom") } links { site("https://spring.io/projects/spring-framework") @@ -2162,7 +2227,7 @@ bom { releaseNotes("https://github.com/spring-projects/spring-framework/releases/tag/v{version}") } } - library("Spring GraphQL", "1.3.4") { + library("Spring GraphQL", "1.4.0-M1") { considerSnapshots() group("org.springframework.graphql") { modules = [ @@ -2180,7 +2245,11 @@ bom { releaseNotes("https://github.com/spring-projects/spring-graphql/releases/tag/v{version}") } } - library("Spring HATEOAS", "2.4.1") { + library("Spring HATEOAS", "2.5.0-M1") { + prohibit { + versionRange "[3.0.0-M1,)" + because "it exceeds our baseline" + } considerSnapshots() group("org.springframework.hateoas") { modules = [ @@ -2197,12 +2266,10 @@ bom { releaseNotes("https://github.com/spring-projects/spring-hateoas/releases/tag/{version}") } } - library("Spring Integration", "6.4.3") { + library("Spring Integration", "6.5.0-M3") { considerSnapshots() group("org.springframework.integration") { - imports = [ - "spring-integration-bom" - ] + bom("spring-integration-bom") } links { site("https://spring.io/projects/spring-integration") @@ -2232,7 +2299,7 @@ bom { releaseNotes("https://github.com/spring-projects/spring-kafka/releases/tag/v{version}") } } - library("Spring LDAP", "3.2.11") { + library("Spring LDAP", "3.3.0-M3") { considerSnapshots() group("org.springframework.ldap") { modules = [ @@ -2255,9 +2322,7 @@ bom { library("Spring Pulsar", "1.2.4") { considerSnapshots() group("org.springframework.pulsar") { - imports = [ - "spring-pulsar-bom" - ] + bom("spring-pulsar-bom") } links { site("https://spring.io/projects/spring-pulsar") @@ -2272,9 +2337,7 @@ bom { library("Spring RESTDocs", "3.0.3") { considerSnapshots() group("org.springframework.restdocs") { - imports = [ - "spring-restdocs-bom" - ] + bom("spring-restdocs-bom") } links { site("https://spring.io/projects/spring-restdocs") @@ -2299,12 +2362,10 @@ bom { releaseNotes("https://github.com/spring-projects/spring-retry/releases/tag/v{version}") } } - library("Spring Security", "6.4.4") { + library("Spring Security", "6.5.0-M3") { considerSnapshots() group("org.springframework.security") { - imports = [ - "spring-security-bom" - ] + bom("spring-security-bom") } links { site("https://spring.io/projects/spring-security") @@ -2322,10 +2383,12 @@ bom { startsWith(["Apple-", "Bean-", "Corn-", "Dragonfruit-"]) because "Spring Session switched to numeric version numbers" } + prohibit { + versionRange "[2020.0.0-M1,)" + because "Spring Session stopped using calver" + } group("org.springframework.session") { - imports = [ - "spring-session-bom" - ] + bom("spring-session-bom") } links { site("https://spring.io/projects/spring-session") @@ -2337,12 +2400,10 @@ bom { releaseNotes("https://github.com/spring-projects/spring-session/releases/tag/{version}") } } - library("Spring WS", "4.0.12") { + library("Spring WS", "4.1.0-M1") { considerSnapshots() group("org.springframework.ws") { - imports = [ - "spring-ws-bom" - ] + bom("spring-ws-bom") } links("spring-webservices") { site("https://spring.io/projects/spring-ws") @@ -2354,7 +2415,7 @@ bom { releaseNotes("https://github.com/spring-projects/spring-ws/releases/tag/v{version}") } } - library("SQLite JDBC", "3.47.2.0") { + library("SQLite JDBC", "3.49.1.0") { group("org.xerial") { modules = [ "sqlite-jdbc" @@ -2367,9 +2428,7 @@ bom { } library("Testcontainers", "1.20.6") { group("org.testcontainers") { - imports = [ - "testcontainers-bom" - ] + bom("testcontainers-bom") } links { site("https://java.testcontainers.org") @@ -2416,7 +2475,7 @@ bom { ] } } - library("Thymeleaf Layout Dialect", "3.3.0") { + library("Thymeleaf Layout Dialect", "3.4.0") { group("nz.net.ultraq.thymeleaf") { modules = [ "thymeleaf-layout-dialect" @@ -2427,6 +2486,10 @@ bom { } } library("Tomcat", "${tomcatVersion}") { + prohibit { + versionRange "[11.0.0-M1,)" + because "it exceeds our Jakarte EE 10 baseline" + } group("org.apache.tomcat") { modules = [ "tomcat-annotations-api", @@ -2449,7 +2512,7 @@ bom { releaseNotes(version -> "https://tomcat.apache.org/tomcat-%s.%s-doc/changelog.html".formatted(version.major(), version.minor())) } } - library("UnboundID LDAPSDK", "6.0.11") { + library("UnboundID LDAPSDK", "7.0.2") { group("com.unboundid") { modules = [ "unboundid-ldapsdk" @@ -2471,7 +2534,7 @@ bom { releaseNotes("https://github.com/undertow-io/undertow/releases/tag/{version}") } } - library("Versions Maven Plugin", "2.17.1") { + library("Versions Maven Plugin", "2.18.0") { group("org.codehaus.mojo") { plugins = [ "versions-maven-plugin" @@ -2481,6 +2544,14 @@ bom { releaseNotes("https://github.com/mojohaus/versions/releases/tag/{version}") } } + library("Vibur", "26.0") { + group("org.vibur") { + modules = [ + "vibur-dbcp", + "vibur-object-pool" + ] + } + } library("WebJars Locator Core", "0.59") { group("org.webjars") { modules = [ @@ -2488,7 +2559,7 @@ bom { ] } } - library("WebJars Locator Lite", "1.0.1") { + library("WebJars Locator Lite", "1.1.0") { group("org.webjars") { modules = [ "webjars-locator-lite" diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/ConditionalOnEnabledDevTools.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/ConditionalOnEnabledDevTools.java new file mode 100644 index 000000000000..c035ae58d26f --- /dev/null +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/ConditionalOnEnabledDevTools.java @@ -0,0 +1,40 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.devtools.autoconfigure; + +import java.lang.annotation.Documented; +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.context.annotation.Conditional; + +/** + * {@link Conditional @Conditional} that matches when DevTools is enabled. + * + * @author Andy Wilkinson + * @since 3.5.0 + */ +@SuppressWarnings("removal") +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.METHOD }) +@Documented +@Conditional(OnEnabledDevToolsCondition.class) +public @interface ConditionalOnEnabledDevTools { + +} diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsDataSourceAutoConfiguration.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsDataSourceAutoConfiguration.java index 24f0a2f80bac..770f88a35101 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsDataSourceAutoConfiguration.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsDataSourceAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -58,7 +58,8 @@ * @since 1.3.3 */ @ConditionalOnClass(DataSource.class) -@Conditional({ OnEnabledDevToolsCondition.class, DevToolsDataSourceCondition.class }) +@ConditionalOnEnabledDevTools +@Conditional(DevToolsDataSourceCondition.class) @AutoConfiguration(after = DataSourceAutoConfiguration.class) @Import(DatabaseShutdownExecutorEntityManagerFactoryDependsOnPostProcessor.class) public class DevToolsDataSourceAutoConfiguration { diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsProperties.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsProperties.java index 2471f19136d1..2a2b4381ab2a 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsProperties.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ * @author Stephane Nicoll * @since 1.3.0 */ -@ConfigurationProperties(prefix = "spring.devtools") +@ConfigurationProperties("spring.devtools") public class DevToolsProperties { private final Restart restart = new Restart(); diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsR2dbcAutoConfiguration.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsR2dbcAutoConfiguration.java index ce5be593743c..7c3e72f650d6 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsR2dbcAutoConfiguration.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/DevToolsR2dbcAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,8 @@ * @since 2.5.6 */ @ConditionalOnClass(ConnectionFactory.class) -@Conditional({ OnEnabledDevToolsCondition.class, DevToolsConnectionFactoryCondition.class }) +@ConditionalOnEnabledDevTools +@Conditional(DevToolsConnectionFactoryCondition.class) @AutoConfiguration(after = R2dbcAutoConfiguration.class) public class DevToolsR2dbcAutoConfiguration { diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/LocalDevToolsAutoConfiguration.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/LocalDevToolsAutoConfiguration.java index 86043b6b6ce7..b0316ec7635b 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/LocalDevToolsAutoConfiguration.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/LocalDevToolsAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,8 +25,8 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.boot.devtools.autoconfigure.DevToolsProperties.Restart; import org.springframework.boot.devtools.classpath.ClassPathChangedEvent; @@ -68,7 +68,7 @@ public class LocalDevToolsAutoConfiguration { * Local LiveReload configuration. */ @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.devtools.livereload", name = "enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.devtools.livereload.enabled", matchIfMissing = true) static class LiveReloadConfiguration { @Bean @@ -96,7 +96,7 @@ LiveReloadServerEventListener liveReloadServerEventListener(OptionalLiveReloadSe */ @Lazy(false) @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.devtools.restart", name = "enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.devtools.restart.enabled", matchIfMissing = true) static class RestartConfiguration { private final DevToolsProperties properties; @@ -134,7 +134,7 @@ FileSystemWatcherFactory fileSystemWatcherFactory() { } @Bean - @ConditionalOnProperty(prefix = "spring.devtools.restart", name = "log-condition-evaluation-delta", + @ConditionalOnBooleanProperty(name = "spring.devtools.restart.log-condition-evaluation-delta", matchIfMissing = true) ConditionEvaluationDeltaLoggingListener conditionEvaluationDeltaLoggingListener() { return new ConditionEvaluationDeltaLoggingListener(); diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/OnEnabledDevToolsCondition.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/OnEnabledDevToolsCondition.java index 1a405ab9931d..bfa767badc4f 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/OnEnabledDevToolsCondition.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/OnEnabledDevToolsCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,7 +28,10 @@ * * @author Madhura Bhave * @since 2.2.0 + * @deprecated since 3.5.0 for removal in 4.0.0 in favor of + * {@link ConditionalOnEnabledDevTools @ConditionalOnEnabledDevTools} */ +@Deprecated(since = "3.5.0", forRemoval = true) public class OnEnabledDevToolsCondition extends SpringBootCondition { @Override diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/RemoteDevToolsAutoConfiguration.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/RemoteDevToolsAutoConfiguration.java index ec8afdbade57..5ecf350ed4da 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/RemoteDevToolsAutoConfiguration.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/RemoteDevToolsAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,6 +24,7 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; @@ -44,7 +45,6 @@ import org.springframework.boot.devtools.restart.server.HttpRestartServerHandler; import org.springframework.boot.devtools.restart.server.SourceDirectoryUrlFilter; import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Conditional; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Import; import org.springframework.core.log.LogMessage; @@ -60,8 +60,8 @@ * @since 1.3.0 */ @AutoConfiguration(after = SecurityAutoConfiguration.class) -@Conditional(OnEnabledDevToolsCondition.class) -@ConditionalOnProperty(prefix = "spring.devtools.remote", name = "secret") +@ConditionalOnEnabledDevTools +@ConditionalOnProperty("spring.devtools.remote.secret") @ConditionalOnClass({ Filter.class, ServerHttpRequest.class }) @Import(RemoteDevtoolsSecurityConfiguration.class) @EnableConfigurationProperties({ ServerProperties.class, DevToolsProperties.class }) @@ -102,7 +102,7 @@ public DispatcherFilter remoteDevToolsDispatcherFilter(AccessManager accessManag * Configuration for remote update and restarts. */ @Configuration(proxyBeanMethods = false) - @ConditionalOnProperty(prefix = "spring.devtools.remote.restart", name = "enabled", matchIfMissing = true) + @ConditionalOnBooleanProperty(name = "spring.devtools.remote.restart.enabled", matchIfMissing = true) static class RemoteRestartConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/TriggerFileFilter.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/TriggerFileFilter.java index d4f197d00b02..1b78252d4145 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/TriggerFileFilter.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/autoconfigure/TriggerFileFilter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -32,7 +32,7 @@ public class TriggerFileFilter implements FileFilter { private final String name; public TriggerFileFilter(String name) { - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); this.name = name; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathChangedEvent.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathChangedEvent.java index b08ce3beff05..5d5c09ebf44c 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathChangedEvent.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathChangedEvent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -46,7 +46,7 @@ public class ClassPathChangedEvent extends ApplicationEvent { */ public ClassPathChangedEvent(Object source, Set changeSet, boolean restartRequired) { super(source); - Assert.notNull(changeSet, "ChangeSet must not be null"); + Assert.notNull(changeSet, "'changeSet' must not be null"); this.changeSet = changeSet; this.restartRequired = restartRequired; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathFileChangeListener.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathFileChangeListener.java index d8d7d0c5b87f..866992a0e4b6 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathFileChangeListener.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathFileChangeListener.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -50,8 +50,8 @@ class ClassPathFileChangeListener implements FileChangeListener { */ ClassPathFileChangeListener(ApplicationEventPublisher eventPublisher, ClassPathRestartStrategy restartStrategy, FileSystemWatcher fileSystemWatcherToStop) { - Assert.notNull(eventPublisher, "EventPublisher must not be null"); - Assert.notNull(restartStrategy, "RestartStrategy must not be null"); + Assert.notNull(eventPublisher, "'eventPublisher' must not be null"); + Assert.notNull(restartStrategy, "'restartStrategy' must not be null"); this.eventPublisher = eventPublisher; this.restartStrategy = restartStrategy; this.fileSystemWatcherToStop = fileSystemWatcherToStop; diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathFileSystemWatcher.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathFileSystemWatcher.java index bf70e37d12a3..f78413dadf46 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathFileSystemWatcher.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/classpath/ClassPathFileSystemWatcher.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -54,8 +54,8 @@ public class ClassPathFileSystemWatcher implements InitializingBean, DisposableB */ public ClassPathFileSystemWatcher(FileSystemWatcherFactory fileSystemWatcherFactory, ClassPathRestartStrategy restartStrategy, URL[] urls) { - Assert.notNull(fileSystemWatcherFactory, "FileSystemWatcherFactory must not be null"); - Assert.notNull(urls, "Urls must not be null"); + Assert.notNull(fileSystemWatcherFactory, "'fileSystemWatcherFactory' must not be null"); + Assert.notNull(urls, "'urls' must not be null"); this.fileSystemWatcher = fileSystemWatcherFactory.getFileSystemWatcher(); this.restartStrategy = restartStrategy; this.fileSystemWatcher.addSourceDirectories(new ClassPathDirectories(urls)); diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/ChangedFile.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/ChangedFile.java index f4cae7d659a9..1313c456db58 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/ChangedFile.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/ChangedFile.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,9 +43,9 @@ public final class ChangedFile { * @param type the type of change */ public ChangedFile(File sourceDirectory, File file, Type type) { - Assert.notNull(sourceDirectory, "SourceDirectory must not be null"); - Assert.notNull(file, "File must not be null"); - Assert.notNull(type, "Type must not be null"); + Assert.notNull(sourceDirectory, "'sourceDirectory' must not be null"); + Assert.notNull(file, "'file' must not be null"); + Assert.notNull(type, "'type' must not be null"); this.sourceDirectory = sourceDirectory; this.file = file; this.type = type; diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/DirectorySnapshot.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/DirectorySnapshot.java index 899714d82a3e..20a547637094 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/DirectorySnapshot.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/DirectorySnapshot.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -50,8 +50,8 @@ class DirectorySnapshot { * @param directory the source directory */ DirectorySnapshot(File directory) { - Assert.notNull(directory, "Directory must not be null"); - Assert.isTrue(!directory.isFile(), () -> "Directory '" + directory + "' must not be a file"); + Assert.notNull(directory, "'directory' must not be null"); + Assert.isTrue(!directory.isFile(), () -> "'directory' [%s] must not be a file".formatted(directory)); this.directory = directory; this.time = new Date(); Set files = new LinkedHashSet<>(); @@ -74,10 +74,10 @@ else if (child.isFile()) { } ChangedFiles getChangedFiles(DirectorySnapshot snapshot, FileFilter triggerFilter) { - Assert.notNull(snapshot, "Snapshot must not be null"); + Assert.notNull(snapshot, "'snapshot' must not be null"); File directory = this.directory; Assert.isTrue(snapshot.directory.equals(directory), - () -> "Snapshot source directory must be '" + directory + "'"); + () -> "'snapshot' source directory must be '" + directory + "'"); Set changes = new LinkedHashSet<>(); Map previousFiles = getFilesMap(); for (FileSnapshot currentFile : snapshot.files) { diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/FileSnapshot.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/FileSnapshot.java index ef29927d11f6..1954c8718cb7 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/FileSnapshot.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/FileSnapshot.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,8 +36,8 @@ class FileSnapshot { private final long lastModified; FileSnapshot(File file) { - Assert.notNull(file, "File must not be null"); - Assert.isTrue(file.isFile() || !file.exists(), "File must not be a directory"); + Assert.notNull(file, "'file' must not be null"); + Assert.isTrue(file.isFile() || !file.exists(), () -> "'file' [%s] must be a normal file".formatted(file)); this.file = file; this.exists = file.exists(); this.length = file.length(); diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/FileSystemWatcher.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/FileSystemWatcher.java index 35a950c6e3b1..876a5bb21798 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/FileSystemWatcher.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/filewatch/FileSystemWatcher.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -95,12 +95,12 @@ public FileSystemWatcher(boolean daemon, Duration pollInterval, Duration quietPe */ public FileSystemWatcher(boolean daemon, Duration pollInterval, Duration quietPeriod, SnapshotStateRepository snapshotStateRepository) { - Assert.notNull(pollInterval, "PollInterval must not be null"); - Assert.notNull(quietPeriod, "QuietPeriod must not be null"); - Assert.isTrue(pollInterval.toMillis() > 0, "PollInterval must be positive"); - Assert.isTrue(quietPeriod.toMillis() > 0, "QuietPeriod must be positive"); + Assert.notNull(pollInterval, "'pollInterval' must not be null"); + Assert.notNull(quietPeriod, "'quietPeriod' must not be null"); + Assert.isTrue(pollInterval.toMillis() > 0, "'pollInterval' must be positive"); + Assert.isTrue(quietPeriod.toMillis() > 0, "'quietPeriod' must be positive"); Assert.isTrue(pollInterval.toMillis() > quietPeriod.toMillis(), - "PollInterval must be greater than QuietPeriod"); + "'pollInterval' must be greater than QuietPeriod"); this.daemon = daemon; this.pollInterval = pollInterval.toMillis(); this.quietPeriod = quietPeriod.toMillis(); @@ -114,7 +114,7 @@ public FileSystemWatcher(boolean daemon, Duration pollInterval, Duration quietPe * @param fileChangeListener the listener to add */ public void addListener(FileChangeListener fileChangeListener) { - Assert.notNull(fileChangeListener, "FileChangeListener must not be null"); + Assert.notNull(fileChangeListener, "'fileChangeListener' must not be null"); synchronized (this.monitor) { checkNotStarted(); this.listeners.add(fileChangeListener); @@ -127,7 +127,7 @@ public void addListener(FileChangeListener fileChangeListener) { * @param directories the directories to monitor */ public void addSourceDirectories(Iterable directories) { - Assert.notNull(directories, "Directories must not be null"); + Assert.notNull(directories, "'directories' must not be null"); synchronized (this.monitor) { directories.forEach(this::addSourceDirectory); } @@ -139,8 +139,8 @@ public void addSourceDirectories(Iterable directories) { * @param directory the directory to monitor */ public void addSourceDirectory(File directory) { - Assert.notNull(directory, "Directory must not be null"); - Assert.isTrue(!directory.isFile(), () -> "Directory '" + directory + "' must not be a file"); + Assert.notNull(directory, "'directory' must not be null"); + Assert.isTrue(!directory.isFile(), () -> "'directory' [%s] must not be a file".formatted(directory)); synchronized (this.monitor) { checkNotStarted(); this.directories.put(directory, null); diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/livereload/Frame.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/livereload/Frame.java index 786a14c42fff..61376ea76230 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/livereload/Frame.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/livereload/Frame.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,13 +40,13 @@ class Frame { * @param payload the text payload */ Frame(String payload) { - Assert.notNull(payload, "Payload must not be null"); + Assert.notNull(payload, "'payload' must not be null"); this.type = Type.TEXT; this.payload = payload.getBytes(); } Frame(Type type) { - Assert.notNull(type, "Type must not be null"); + Assert.notNull(type, "'type' must not be null"); this.type = type; this.payload = NO_BYTES; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/client/ClassPathChangeUploader.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/client/ClassPathChangeUploader.java index 44bf7d24d58d..5f2fea3db0c0 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/client/ClassPathChangeUploader.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/client/ClassPathChangeUploader.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -76,8 +76,8 @@ public class ClassPathChangeUploader implements ApplicationListener mappers; public Dispatcher(AccessManager accessManager, Collection mappers) { - Assert.notNull(accessManager, "AccessManager must not be null"); - Assert.notNull(mappers, "Mappers must not be null"); + Assert.notNull(accessManager, "'accessManager' must not be null"); + Assert.notNull(mappers, "'mappers' must not be null"); this.accessManager = accessManager; this.mappers = new ArrayList<>(mappers); AnnotationAwareOrderComparator.sort(this.mappers); diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/DispatcherFilter.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/DispatcherFilter.java index 351f56896658..f8a2fa7db1ef 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/DispatcherFilter.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/DispatcherFilter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,7 +45,7 @@ public class DispatcherFilter implements Filter { private final Dispatcher dispatcher; public DispatcherFilter(Dispatcher dispatcher) { - Assert.notNull(dispatcher, "Dispatcher must not be null"); + Assert.notNull(dispatcher, "'dispatcher' must not be null"); this.dispatcher = dispatcher; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/HttpHeaderAccessManager.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/HttpHeaderAccessManager.java index c04366f8c6d8..6c3a3d49966c 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/HttpHeaderAccessManager.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/HttpHeaderAccessManager.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,8 +33,8 @@ public class HttpHeaderAccessManager implements AccessManager { private final String expectedSecret; public HttpHeaderAccessManager(String headerName, String expectedSecret) { - Assert.hasLength(headerName, "HeaderName must not be empty"); - Assert.hasLength(expectedSecret, "ExpectedSecret must not be empty"); + Assert.hasLength(headerName, "'headerName' must not be empty"); + Assert.hasLength(expectedSecret, "'expectedSecret' must not be empty"); this.headerName = headerName; this.expectedSecret = expectedSecret; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/HttpStatusHandler.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/HttpStatusHandler.java index 218e6796e003..67bd888ad2dc 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/HttpStatusHandler.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/HttpStatusHandler.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ public HttpStatusHandler() { * @param status the status */ public HttpStatusHandler(HttpStatus status) { - Assert.notNull(status, "Status must not be null"); + Assert.notNull(status, "'status' must not be null"); this.status = status; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/UrlHandlerMapper.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/UrlHandlerMapper.java index 3237ca00ba61..2bd1094ef323 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/UrlHandlerMapper.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/remote/server/UrlHandlerMapper.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,8 +38,8 @@ public class UrlHandlerMapper implements HandlerMapper { * @param handler the handler to use */ public UrlHandlerMapper(String url, Handler handler) { - Assert.hasLength(url, "URL must not be empty"); - Assert.isTrue(url.startsWith("/"), "URL must start with '/'"); + Assert.hasLength(url, "'url' must not be empty"); + Assert.isTrue(url.startsWith("/"), "'url' must start with '/'"); this.requestUri = url; this.handler = handler; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/MainMethod.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/MainMethod.java index 9f3767ae6b94..c02b2dc77fe2 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/MainMethod.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/MainMethod.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ class MainMethod { } MainMethod(Thread thread) { - Assert.notNull(thread, "Thread must not be null"); + Assert.notNull(thread, "'thread' must not be null"); this.method = getMainMethod(thread); } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/Restarter.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/Restarter.java index 4d71ceddd127..16b1609060d5 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/Restarter.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/Restarter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -129,9 +129,9 @@ public class Restarter { * @see #initialize(String[]) */ protected Restarter(Thread thread, String[] args, boolean forceReferenceCleanup, RestartInitializer initializer) { - Assert.notNull(thread, "Thread must not be null"); - Assert.notNull(args, "Args must not be null"); - Assert.notNull(initializer, "Initializer must not be null"); + Assert.notNull(thread, "'thread' must not be null"); + Assert.notNull(args, "'args' must not be null"); + Assert.notNull(initializer, "'initializer' must not be null"); if (this.logger.isDebugEnabled()) { this.logger.debug("Creating new Restarter for thread " + thread); } @@ -208,7 +208,7 @@ private void setEnabled(boolean enabled) { * @param urls the urls to add */ public void addUrls(Collection urls) { - Assert.notNull(urls, "Urls must not be null"); + Assert.notNull(urls, "'urls' must not be null"); this.urls.addAll(urls); } @@ -217,7 +217,7 @@ public void addUrls(Collection urls) { * @param classLoaderFiles the files to add */ public void addClassLoaderFiles(ClassLoaderFiles classLoaderFiles) { - Assert.notNull(classLoaderFiles, "ClassLoaderFiles must not be null"); + Assert.notNull(classLoaderFiles, "'classLoaderFiles' must not be null"); this.classLoaderFiles.addAll(classLoaderFiles); } @@ -272,7 +272,7 @@ protected void start(FailureHandler failureHandler) throws Exception { } private Throwable doStart() throws Exception { - Assert.notNull(this.mainClassName, "Unable to find the main class to restart"); + Assert.state(this.mainClassName != null, "Unable to find the main class to restart"); URL[] urls = this.urls.toArray(new URL[0]); ClassLoaderFiles updatedFiles = new ClassLoaderFiles(this.classLoaderFiles); ClassLoader classLoader = new RestartClassLoader(this.applicationClassLoader, urls, updatedFiles); diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFile.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFile.java index 46df7df0d325..39b44b8d233c 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFile.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFile.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -54,12 +54,12 @@ public ClassLoaderFile(Kind kind, byte[] contents) { * @param contents the file contents */ public ClassLoaderFile(Kind kind, long lastModified, byte[] contents) { - Assert.notNull(kind, "Kind must not be null"); + Assert.notNull(kind, "'kind' must not be null"); if (kind == Kind.DELETED) { - Assert.isTrue(contents == null, "Contents must be null"); + Assert.isTrue(contents == null, "'contents' must be null"); } else { - Assert.isTrue(contents != null, "Contents must not be null"); + Assert.isTrue(contents != null, "'contents' must not be null"); } this.kind = kind; this.lastModified = lastModified; diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFiles.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFiles.java index 06e7d8f3f97b..3d0c0708d9f9 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFiles.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFiles.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -55,7 +55,7 @@ public ClassLoaderFiles() { * @param classLoaderFiles the source classloader files. */ public ClassLoaderFiles(ClassLoaderFiles classLoaderFiles) { - Assert.notNull(classLoaderFiles, "ClassLoaderFiles must not be null"); + Assert.notNull(classLoaderFiles, "'classLoaderFiles' must not be null"); this.sourceDirectories = new LinkedHashMap<>(classLoaderFiles.sourceDirectories); } @@ -65,7 +65,7 @@ public ClassLoaderFiles(ClassLoaderFiles classLoaderFiles) { * @param files the files to add */ public void addAll(ClassLoaderFiles files) { - Assert.notNull(files, "Files must not be null"); + Assert.notNull(files, "'files' must not be null"); for (SourceDirectory directory : files.getSourceDirectories()) { for (Map.Entry entry : directory.getFilesEntrySet()) { addFile(directory.getName(), entry.getKey(), entry.getValue()); @@ -89,9 +89,9 @@ public void addFile(String name, ClassLoaderFile file) { * @param file the file to add */ public void addFile(String sourceDirectory, String name, ClassLoaderFile file) { - Assert.notNull(sourceDirectory, "SourceDirectory must not be null"); - Assert.notNull(name, "Name must not be null"); - Assert.notNull(file, "File must not be null"); + Assert.notNull(sourceDirectory, "'sourceDirectory' must not be null"); + Assert.notNull(name, "'name' must not be null"); + Assert.notNull(file, "'file' must not be null"); removeAll(name); getOrCreateSourceDirectory(sourceDirectory).add(name, file); } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/RestartClassLoader.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/RestartClassLoader.java index 94f1a10143fd..03599d099f8e 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/RestartClassLoader.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/classloader/RestartClassLoader.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -57,8 +57,8 @@ public RestartClassLoader(ClassLoader parent, URL[] urls) { */ public RestartClassLoader(ClassLoader parent, URL[] urls, ClassLoaderFileRepository updatedFiles) { super(urls, parent); - Assert.notNull(parent, "Parent must not be null"); - Assert.notNull(updatedFiles, "UpdatedFiles must not be null"); + Assert.notNull(parent, "'parent' must not be null"); + Assert.notNull(updatedFiles, "'updatedFiles' must not be null"); this.updatedFiles = updatedFiles; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/HttpRestartServer.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/HttpRestartServer.java index e2cbaeeac5b8..5b2c99c5e1ea 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/HttpRestartServer.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/HttpRestartServer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,7 +48,7 @@ public class HttpRestartServer { * the local classpath */ public HttpRestartServer(SourceDirectoryUrlFilter sourceDirectoryUrlFilter) { - Assert.notNull(sourceDirectoryUrlFilter, "SourceDirectoryUrlFilter must not be null"); + Assert.notNull(sourceDirectoryUrlFilter, "'sourceDirectoryUrlFilter' must not be null"); this.server = new RestartServer(sourceDirectoryUrlFilter); } @@ -57,7 +57,7 @@ public HttpRestartServer(SourceDirectoryUrlFilter sourceDirectoryUrlFilter) { * @param restartServer the underlying restart server */ public HttpRestartServer(RestartServer restartServer) { - Assert.notNull(restartServer, "RestartServer must not be null"); + Assert.notNull(restartServer, "'restartServer' must not be null"); this.server = restartServer; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/HttpRestartServerHandler.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/HttpRestartServerHandler.java index 66100c1f3dff..d62dbf9d5e86 100644 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/HttpRestartServerHandler.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/HttpRestartServerHandler.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ public class HttpRestartServerHandler implements Handler { * @param server the server to adapt */ public HttpRestartServerHandler(HttpRestartServer server) { - Assert.notNull(server, "Server must not be null"); + Assert.notNull(server, "'server' must not be null"); this.server = server; } diff --git a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/RestartServer.java b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/RestartServer.java index a88526fc8dcb..c3cc42b62d90 100755 --- a/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/RestartServer.java +++ b/spring-boot-project/spring-boot-devtools/src/main/java/org/springframework/boot/devtools/restart/server/RestartServer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -68,8 +68,8 @@ public RestartServer(SourceDirectoryUrlFilter sourceDirectoryUrlFilter) { * @param classLoader the application classloader */ public RestartServer(SourceDirectoryUrlFilter sourceDirectoryUrlFilter, ClassLoader classLoader) { - Assert.notNull(sourceDirectoryUrlFilter, "SourceDirectoryUrlFilter must not be null"); - Assert.notNull(classLoader, "ClassLoader must not be null"); + Assert.notNull(sourceDirectoryUrlFilter, "'sourceDirectoryUrlFilter' must not be null"); + Assert.notNull(classLoader, "'classLoader' must not be null"); this.sourceDirectoryUrlFilter = sourceDirectoryUrlFilter; this.classLoader = classLoader; } diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/autoconfigure/OnEnabledDevToolsConditionTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/autoconfigure/OnEnabledDevToolsConditionTests.java index c0047b93ad35..adc80b11107c 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/autoconfigure/OnEnabledDevToolsConditionTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/autoconfigure/OnEnabledDevToolsConditionTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -65,6 +65,7 @@ void outcomeWhenDevtoolsShouldBeEnabledIsFalseShouldNotMatch() { static class TestConfiguration { @Bean + @SuppressWarnings("removal") @Conditional(OnEnabledDevToolsCondition.class) String test() { return "hello"; diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/autoconfigure/TriggerFileFilterTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/autoconfigure/TriggerFileFilterTests.java index 458a5826d517..56d713291e36 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/autoconfigure/TriggerFileFilterTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/autoconfigure/TriggerFileFilterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ class TriggerFileFilterTests { @Test void nameMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new TriggerFileFilter(null)) - .withMessageContaining("Name must not be null"); + .withMessageContaining("'name' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathChangedEventTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathChangedEventTests.java index ee62e930858b..c4b78d9dda14 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathChangedEventTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathChangedEventTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ class ClassPathChangedEventTests { @Test void changeSetMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new ClassPathChangedEvent(this.source, null, false)) - .withMessageContaining("ChangeSet must not be null"); + .withMessageContaining("'changeSet' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathFileChangeListenerTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathFileChangeListenerTests.java index 29ae6eed8b56..8f0d5c0324b4 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathFileChangeListenerTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathFileChangeListenerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -59,14 +59,14 @@ class ClassPathFileChangeListenerTests { void eventPublisherMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new ClassPathFileChangeListener(null, this.restartStrategy, this.fileSystemWatcher)) - .withMessageContaining("EventPublisher must not be null"); + .withMessageContaining("'eventPublisher' must not be null"); } @Test void restartStrategyMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new ClassPathFileChangeListener(this.eventPublisher, null, this.fileSystemWatcher)) - .withMessageContaining("RestartStrategy must not be null"); + .withMessageContaining("'restartStrategy' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathFileSystemWatcherTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathFileSystemWatcherTests.java index 877374e93dd3..dd883fa3b197 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathFileSystemWatcherTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/classpath/ClassPathFileSystemWatcherTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -55,7 +55,7 @@ void urlsMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new ClassPathFileSystemWatcher(mock(FileSystemWatcherFactory.class), mock(ClassPathRestartStrategy.class), (URL[]) null)) - .withMessageContaining("Urls must not be null"); + .withMessageContaining("'urls' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/ChangedFileTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/ChangedFileTests.java index 7a59b90c06db..dd827beea12d 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/ChangedFileTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/ChangedFileTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,14 +40,14 @@ class ChangedFileTests { void sourceDirectoryMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new ChangedFile(null, new File(this.tempDir, "file"), Type.ADD)) - .withMessageContaining("SourceDirectory must not be null"); + .withMessageContaining("'sourceDirectory' must not be null"); } @Test void fileMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new ChangedFile(new File(this.tempDir, "directory"), null, Type.ADD)) - .withMessageContaining("File must not be null"); + .withMessageContaining("'file' must not be null"); } @Test @@ -55,7 +55,7 @@ void typeMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy( () -> new ChangedFile(new File(this.tempDir, "file"), new File(this.tempDir, "directory"), null)) - .withMessageContaining("Type must not be null"); + .withMessageContaining("'type' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/DirectorySnapshotTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/DirectorySnapshotTests.java index c4c52da7ceb6..96e3a721bb42 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/DirectorySnapshotTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/DirectorySnapshotTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,7 +53,7 @@ void setup() throws Exception { @Test void directoryMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new DirectorySnapshot(null)) - .withMessageContaining("Directory must not be null"); + .withMessageContaining("'directory' must not be null"); } @Test @@ -61,7 +61,7 @@ void directoryMustNotBeFile() throws Exception { File file = new File(this.tempDir, "file"); file.createNewFile(); assertThatIllegalArgumentException().isThrownBy(() -> new DirectorySnapshot(file)) - .withMessageContaining("Directory '" + file + "' must not be a file"); + .withMessageContaining("'directory' [" + file + "] must not be a file"); } @Test @@ -103,14 +103,14 @@ void notEqualsWhenAFileIsModified() throws Exception { @Test void getChangedFilesSnapshotMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> this.initialSnapshot.getChangedFiles(null, null)) - .withMessageContaining("Snapshot must not be null"); + .withMessageContaining("'snapshot' must not be null"); } @Test void getChangedFilesSnapshotMustBeTheSameSourceDirectory() { assertThatIllegalArgumentException().isThrownBy( () -> this.initialSnapshot.getChangedFiles(new DirectorySnapshot(createTestDirectoryStructure()), null)) - .withMessageContaining("Snapshot source directory must be '" + this.directory + "'"); + .withMessageContaining("'snapshot' source directory must be '" + this.directory + "'"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/FileSnapshotTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/FileSnapshotTests.java index 72705e1dd219..48588899f5aa 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/FileSnapshotTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/FileSnapshotTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ class FileSnapshotTests { @Test void fileMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new FileSnapshot(null)) - .withMessageContaining("File must not be null"); + .withMessageContaining("'file' must not be null"); } @Test @@ -55,7 +55,7 @@ void fileMustNotBeADirectory() { File file = new File(this.tempDir, "file"); file.mkdir(); assertThatIllegalArgumentException().isThrownBy(() -> new FileSnapshot(file)) - .withMessageContaining("File must not be a directory"); + .withMessageContaining("'file' [" + file + "] must be a normal file"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/FileSystemWatcherTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/FileSystemWatcherTests.java index 612850f90f64..c8b1c08e7294 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/FileSystemWatcherTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/filewatch/FileSystemWatcherTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -64,27 +64,27 @@ void setup() { void pollIntervalMustBePositive() { assertThatIllegalArgumentException() .isThrownBy(() -> new FileSystemWatcher(true, Duration.ofMillis(0), Duration.ofMillis(1))) - .withMessageContaining("PollInterval must be positive"); + .withMessageContaining("'pollInterval' must be positive"); } @Test void quietPeriodMustBePositive() { assertThatIllegalArgumentException() .isThrownBy(() -> new FileSystemWatcher(true, Duration.ofMillis(1), Duration.ofMillis(0))) - .withMessageContaining("QuietPeriod must be positive"); + .withMessageContaining("'quietPeriod' must be positive"); } @Test void pollIntervalMustBeGreaterThanQuietPeriod() { assertThatIllegalArgumentException() .isThrownBy(() -> new FileSystemWatcher(true, Duration.ofMillis(1), Duration.ofMillis(1))) - .withMessageContaining("PollInterval must be greater than QuietPeriod"); + .withMessageContaining("'pollInterval' must be greater than QuietPeriod"); } @Test void listenerMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> this.watcher.addListener(null)) - .withMessageContaining("FileChangeListener must not be null"); + .withMessageContaining("'fileChangeListener' must not be null"); } @Test @@ -97,7 +97,7 @@ void cannotAddListenerToStartedListener() { @Test void sourceDirectoryMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> this.watcher.addSourceDirectory(null)) - .withMessageContaining("Directory must not be null"); + .withMessageContaining("'directory' must not be null"); } @Test @@ -106,7 +106,7 @@ void sourceDirectoryMustNotBeAFile() throws IOException { assertThat(file.createNewFile()).isTrue(); assertThat(file).isFile(); assertThatIllegalArgumentException().isThrownBy(() -> this.watcher.addSourceDirectory(file)) - .withMessageContaining("Directory '" + file + "' must not be a file"); + .withMessageContaining("'directory' [" + file + "] must not be a file"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/livereload/FrameTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/livereload/FrameTests.java index d54764ffef8a..76dd1dc691fc 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/livereload/FrameTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/livereload/FrameTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,13 +36,13 @@ class FrameTests { @Test void payloadMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new Frame((String) null)) - .withMessageContaining("Payload must not be null"); + .withMessageContaining("'payload' must not be null"); } @Test void typeMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new Frame((Frame.Type) null)) - .withMessageContaining("Type must not be null"); + .withMessageContaining("'type' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/ClassPathChangeUploaderTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/ClassPathChangeUploaderTests.java index 7b941938d183..bf8cb73e015b 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/ClassPathChangeUploaderTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/ClassPathChangeUploaderTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -67,20 +67,20 @@ void setup() { @Test void urlMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new ClassPathChangeUploader(null, this.requestFactory)) - .withMessageContaining("URL must not be empty"); + .withMessageContaining("'url' must not be empty"); } @Test void urlMustNotBeEmpty() { assertThatIllegalArgumentException().isThrownBy(() -> new ClassPathChangeUploader("", this.requestFactory)) - .withMessageContaining("URL must not be empty"); + .withMessageContaining("'url' must not be empty"); } @Test void requestFactoryMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new ClassPathChangeUploader("http://localhost:8080", null)) - .withMessageContaining("RequestFactory must not be null"); + .withMessageContaining("'requestFactory' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/DelayedLiveReloadTriggerTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/DelayedLiveReloadTriggerTests.java index df10274811c1..e6a30590ca61 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/DelayedLiveReloadTriggerTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/DelayedLiveReloadTriggerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -77,28 +77,28 @@ void setup() { void liveReloadServerMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new DelayedLiveReloadTrigger(null, this.requestFactory, URL)) - .withMessageContaining("LiveReloadServer must not be null"); + .withMessageContaining("'liveReloadServer' must not be null"); } @Test void requestFactoryMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new DelayedLiveReloadTrigger(this.liveReloadServer, null, URL)) - .withMessageContaining("RequestFactory must not be null"); + .withMessageContaining("'requestFactory' must not be null"); } @Test void urlMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new DelayedLiveReloadTrigger(this.liveReloadServer, this.requestFactory, null)) - .withMessageContaining("URL must not be empty"); + .withMessageContaining("'url' must not be empty"); } @Test void urlMustNotBeEmpty() { assertThatIllegalArgumentException() .isThrownBy(() -> new DelayedLiveReloadTrigger(this.liveReloadServer, this.requestFactory, "")) - .withMessageContaining("URL must not be empty"); + .withMessageContaining("'url' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/HttpHeaderInterceptorTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/HttpHeaderInterceptorTests.java index 5140994e328f..0a474a6272ec 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/HttpHeaderInterceptorTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/client/HttpHeaderInterceptorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -74,25 +74,25 @@ void setup() { @Test void constructorNullHeaderName() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpHeaderInterceptor(null, this.value)) - .withMessageContaining("Name must not be empty"); + .withMessageContaining("'name' must not be empty"); } @Test void constructorEmptyHeaderName() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpHeaderInterceptor("", this.value)) - .withMessageContaining("Name must not be empty"); + .withMessageContaining("'name' must not be empty"); } @Test void constructorNullHeaderValue() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpHeaderInterceptor(this.name, null)) - .withMessageContaining("Value must not be empty"); + .withMessageContaining("'value' must not be empty"); } @Test void constructorEmptyHeaderValue() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpHeaderInterceptor(this.name, "")) - .withMessageContaining("Value must not be empty"); + .withMessageContaining("'value' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/DispatcherFilterTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/DispatcherFilterTests.java index b684e14178b2..a3bb24216c8c 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/DispatcherFilterTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/DispatcherFilterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -66,7 +66,7 @@ void setup() { @Test void dispatcherMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new DispatcherFilter(null)) - .withMessageContaining("Dispatcher must not be null"); + .withMessageContaining("'dispatcher' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/DispatcherTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/DispatcherTests.java index 45be879837c3..2178e6b00aa3 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/DispatcherTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/DispatcherTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -63,13 +63,13 @@ class DispatcherTests { @Test void accessManagerMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new Dispatcher(null, Collections.emptyList())) - .withMessageContaining("AccessManager must not be null"); + .withMessageContaining("'accessManager' must not be null"); } @Test void mappersMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new Dispatcher(this.accessManager, null)) - .withMessageContaining("Mappers must not be null"); + .withMessageContaining("'mappers' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/HttpHeaderAccessManagerTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/HttpHeaderAccessManagerTests.java index 1a234cad6ef5..b4c0a4f5df9c 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/HttpHeaderAccessManagerTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/HttpHeaderAccessManagerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -54,25 +54,25 @@ void setup() { @Test void headerNameMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpHeaderAccessManager(null, SECRET)) - .withMessageContaining("HeaderName must not be empty"); + .withMessageContaining("'headerName' must not be empty"); } @Test void headerNameMustNotBeEmpty() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpHeaderAccessManager("", SECRET)) - .withMessageContaining("HeaderName must not be empty"); + .withMessageContaining("'headerName' must not be empty"); } @Test void expectedSecretMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpHeaderAccessManager(HEADER, null)) - .withMessageContaining("ExpectedSecret must not be empty"); + .withMessageContaining("'expectedSecret' must not be empty"); } @Test void expectedSecretMustNotBeEmpty() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpHeaderAccessManager(HEADER, "")) - .withMessageContaining("ExpectedSecret must not be empty"); + .withMessageContaining("'expectedSecret' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/HttpStatusHandlerTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/HttpStatusHandlerTests.java index 4adef3962403..4de622902c08 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/HttpStatusHandlerTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/HttpStatusHandlerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,7 +56,7 @@ void setup() { @Test void statusMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpStatusHandler(null)) - .withMessageContaining("Status must not be null"); + .withMessageContaining("'status' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/UrlHandlerMapperTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/UrlHandlerMapperTests.java index 7febedbb1777..cc4f675b5c7b 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/UrlHandlerMapperTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/remote/server/UrlHandlerMapperTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,19 +40,19 @@ class UrlHandlerMapperTests { @Test void requestUriMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new UrlHandlerMapper(null, this.handler)) - .withMessageContaining("URL must not be empty"); + .withMessageContaining("'url' must not be empty"); } @Test void requestUriMustNotBeEmpty() { assertThatIllegalArgumentException().isThrownBy(() -> new UrlHandlerMapper("", this.handler)) - .withMessageContaining("URL must not be empty"); + .withMessageContaining("'url' must not be empty"); } @Test void requestUrlMustStartWithSlash() { assertThatIllegalArgumentException().isThrownBy(() -> new UrlHandlerMapper("tunnel", this.handler)) - .withMessageContaining("URL must start with '/'"); + .withMessageContaining("'url' must start with '/'"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/MainMethodTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/MainMethodTests.java index 758c3f95c7ef..5ea21115cbec 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/MainMethodTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/MainMethodTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ void setup() throws Exception { @Test void threadMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new MainMethod(null)) - .withMessageContaining("Thread must not be null"); + .withMessageContaining("'thread' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/RestarterTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/RestarterTests.java index ed1dd1bb3a17..f325edd88fe4 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/RestarterTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/RestarterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -113,7 +113,7 @@ void getOrAddAttributeWithNewAttribute() { @Test void addUrlsMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> Restarter.getInstance().addUrls(null)) - .withMessageContaining("Urls must not be null"); + .withMessageContaining("'urls' must not be null"); } @Test @@ -130,7 +130,7 @@ void addUrls() throws Exception { @Test void addClassLoaderFilesMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> Restarter.getInstance().addClassLoaderFiles(null)) - .withMessageContaining("ClassLoaderFiles must not be null"); + .withMessageContaining("'classLoaderFiles' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFileTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFileTests.java index db89e0b2d77d..286781ae4876 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFileTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFileTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,25 +35,25 @@ class ClassLoaderFileTests { @Test void kindMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new ClassLoaderFile(null, null)) - .withMessageContaining("Kind must not be null"); + .withMessageContaining("'kind' must not be null"); } @Test void addedContentsMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new ClassLoaderFile(Kind.ADDED, null)) - .withMessageContaining("Contents must not be null"); + .withMessageContaining("'contents' must not be null"); } @Test void modifiedContentsMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new ClassLoaderFile(Kind.MODIFIED, null)) - .withMessageContaining("Contents must not be null"); + .withMessageContaining("'contents' must not be null"); } @Test void deletedContentsMustBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new ClassLoaderFile(Kind.DELETED, new byte[10])) - .withMessageContaining("Contents must be null"); + .withMessageContaining("'contents' must be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFilesTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFilesTests.java index a965dcefeace..c3903bcec956 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFilesTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/ClassLoaderFilesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,13 +43,13 @@ class ClassLoaderFilesTests { @Test void addFileNameMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> this.files.addFile(null, mock(ClassLoaderFile.class))) - .withMessageContaining("Name must not be null"); + .withMessageContaining("'name' must not be null"); } @Test void addFileFileMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> this.files.addFile("test", null)) - .withMessageContaining("File must not be null"); + .withMessageContaining("'file' must not be null"); } @Test @@ -153,7 +153,7 @@ void getSize() { @Test void classLoaderFilesMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new ClassLoaderFiles(null)) - .withMessageContaining("ClassLoaderFiles must not be null"); + .withMessageContaining("'classLoaderFiles' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/RestartClassLoaderTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/RestartClassLoaderTests.java index 40f9fcc3ac12..612686599a08 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/RestartClassLoaderTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/classloader/RestartClassLoaderTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -104,14 +104,14 @@ private File createSampleJarFile(File tempDir) throws IOException { @Test void parentMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new RestartClassLoader(null, new URL[] {})) - .withMessageContaining("Parent must not be null"); + .withMessageContaining("'parent' must not be null"); } @Test void updatedFilesMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new RestartClassLoader(this.parentClassLoader, new URL[] {}, null)) - .withMessageContaining("UpdatedFiles must not be null"); + .withMessageContaining("'updatedFiles' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/HttpRestartServerHandlerTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/HttpRestartServerHandlerTests.java index 8d0ec158fcf1..905d51868314 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/HttpRestartServerHandlerTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/HttpRestartServerHandlerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ class HttpRestartServerHandlerTests { @Test void serverMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpRestartServerHandler(null)) - .withMessageContaining("Server must not be null"); + .withMessageContaining("'server' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/HttpRestartServerTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/HttpRestartServerTests.java index 1d8842812e1f..2c100127d42d 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/HttpRestartServerTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/HttpRestartServerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -60,13 +60,13 @@ void setup() { @Test void sourceDirectoryUrlFilterMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpRestartServer((SourceDirectoryUrlFilter) null)) - .withMessageContaining("SourceDirectoryUrlFilter must not be null"); + .withMessageContaining("'sourceDirectoryUrlFilter' must not be null"); } @Test void restartServerMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new HttpRestartServer((RestartServer) null)) - .withMessageContaining("RestartServer must not be null"); + .withMessageContaining("'restartServer' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/RestartServerTests.java b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/RestartServerTests.java index 1305a65d480d..728d32a3b90f 100644 --- a/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/RestartServerTests.java +++ b/spring-boot-project/spring-boot-devtools/src/test/java/org/springframework/boot/devtools/restart/server/RestartServerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,7 +45,7 @@ class RestartServerTests { @Test void sourceDirectoryUrlFilterMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new RestartServer((SourceDirectoryUrlFilter) null)) - .withMessageContaining("SourceDirectoryUrlFilter must not be null"); + .withMessageContaining("'sourceDirectoryUrlFilter' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/ldap/LLdapDockerComposeConnectionDetailsFactoryIntegrationTests.java b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/ldap/LLdapDockerComposeConnectionDetailsFactoryIntegrationTests.java new file mode 100644 index 000000000000..eb84a441cd6d --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/ldap/LLdapDockerComposeConnectionDetailsFactoryIntegrationTests.java @@ -0,0 +1,41 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.docker.compose.service.connection.ldap; + +import org.springframework.boot.autoconfigure.ldap.LdapConnectionDetails; +import org.springframework.boot.docker.compose.service.connection.test.DockerComposeTest; +import org.springframework.boot.testsupport.container.TestImage; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Integration tests for {@link LLdapDockerComposeConnectionDetailsFactory}. + * + * @author Eddú Meléndez + */ +class LLdapDockerComposeConnectionDetailsFactoryIntegrationTests { + + @DockerComposeTest(composeFile = "lldap-compose.yaml", image = TestImage.LLDAP) + void runCreatesConnectionDetails(LdapConnectionDetails connectionDetails) { + assertThat(connectionDetails.getUsername()).isEqualTo("cn=admin,ou=people,dc=springframework,dc=org"); + assertThat(connectionDetails.getPassword()).isEqualTo("somepassword"); + assertThat(connectionDetails.getBase()).isEqualTo("dc=springframework,dc=org"); + assertThat(connectionDetails.getUrls()).hasSize(1); + assertThat(connectionDetails.getUrls()[0]).startsWith("ldap://"); + } + +} diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactoryIntegrationTests.java b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactoryIntegrationTests.java index 40405e24c622..430358fe944c 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactoryIntegrationTests.java +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactoryIntegrationTests.java @@ -35,6 +35,7 @@ * @author Andy Wilkinson * @author Phillip Webb * @author Scott Frederick + * @author He Zean */ class PostgresJdbcDockerComposeConnectionDetailsFactoryIntegrationTests { @@ -60,22 +61,38 @@ void runWithBitnamiImageCreatesConnectionDetails(JdbcConnectionDetails connectio checkDatabaseAccess(connectionDetails); } + @DockerComposeTest(composeFile = "postgres-application-name-compose.yaml", image = TestImage.POSTGRESQL) + void runCreatesConnectionDetailsApplicationName(JdbcConnectionDetails connectionDetails) + throws ClassNotFoundException { + assertThat(connectionDetails.getUsername()).isEqualTo("myuser"); + assertThat(connectionDetails.getPassword()).isEqualTo("secret"); + assertThat(connectionDetails.getJdbcUrl()).startsWith("jdbc:postgresql://") + .endsWith("?ApplicationName=spring+boot"); + assertThat(executeQuery(connectionDetails, "select current_setting('application_name')", String.class)) + .isEqualTo("spring boot"); + } + private void assertConnectionDetails(JdbcConnectionDetails connectionDetails) { assertThat(connectionDetails.getUsername()).isEqualTo("myuser"); assertThat(connectionDetails.getPassword()).isEqualTo("secret"); assertThat(connectionDetails.getJdbcUrl()).startsWith("jdbc:postgresql://").endsWith("/mydatabase"); } - @SuppressWarnings("unchecked") private void checkDatabaseAccess(JdbcConnectionDetails connectionDetails) throws ClassNotFoundException { + assertThat(executeQuery(connectionDetails, DatabaseDriver.POSTGRESQL.getValidationQuery(), Integer.class)) + .isEqualTo(1); + } + + @SuppressWarnings("unchecked") + private T executeQuery(JdbcConnectionDetails connectionDetails, String sql, Class resultClass) + throws ClassNotFoundException { SimpleDriverDataSource dataSource = new SimpleDriverDataSource(); dataSource.setUrl(connectionDetails.getJdbcUrl()); dataSource.setUsername(connectionDetails.getUsername()); dataSource.setPassword(connectionDetails.getPassword()); dataSource.setDriverClass((Class) ClassUtils.forName(connectionDetails.getDriverClassName(), getClass().getClassLoader())); - JdbcTemplate template = new JdbcTemplate(dataSource); - assertThat(template.queryForObject(DatabaseDriver.POSTGRESQL.getValidationQuery(), Integer.class)).isEqualTo(1); + return new JdbcTemplate(dataSource).queryForObject(sql, resultClass); } } diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactoryIntegrationTests.java b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactoryIntegrationTests.java index 9d97c3e7abde..d1b5c8bfe024 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactoryIntegrationTests.java +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactoryIntegrationTests.java @@ -20,6 +20,7 @@ import io.r2dbc.spi.ConnectionFactories; import io.r2dbc.spi.ConnectionFactoryOptions; +import io.r2dbc.spi.Option; import org.springframework.boot.autoconfigure.r2dbc.R2dbcConnectionDetails; import org.springframework.boot.docker.compose.service.connection.test.DockerComposeTest; @@ -36,6 +37,7 @@ * @author Andy Wilkinson * @author Phillip Webb * @author Scott Frederick + * @author He Zean */ class PostgresR2dbcDockerComposeConnectionDetailsFactoryIntegrationTests { @@ -62,21 +64,37 @@ void runWithBitnamiImageCreatesConnectionDetails(R2dbcConnectionDetails connecti checkDatabaseAccess(connectionDetails); } + @DockerComposeTest(composeFile = "postgres-application-name-compose.yaml", image = TestImage.POSTGRESQL) + void runCreatesConnectionDetailsApplicationName(R2dbcConnectionDetails connectionDetails) { + assertConnectionDetails(connectionDetails); + ConnectionFactoryOptions options = connectionDetails.getConnectionFactoryOptions(); + assertThat(options.getValue(Option.valueOf("applicationName"))).isEqualTo("spring boot"); + assertThat(executeQuery(connectionDetails, "select current_setting('application_name')", String.class)) + .isEqualTo("spring boot"); + } + private void assertConnectionDetails(R2dbcConnectionDetails connectionDetails) { - ConnectionFactoryOptions connectionFactoryOptions = connectionDetails.getConnectionFactoryOptions(); - assertThat(connectionFactoryOptions.toString()).contains("database=mydatabase", "driver=postgresql", - "password=REDACTED", "user=myuser"); - assertThat(connectionFactoryOptions.getRequiredValue(ConnectionFactoryOptions.PASSWORD)).isEqualTo("secret"); + ConnectionFactoryOptions options = connectionDetails.getConnectionFactoryOptions(); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.HOST)).isNotNull(); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.PORT)).isNotNull(); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.DATABASE)).isEqualTo("mydatabase"); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.USER)).isEqualTo("myuser"); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.PASSWORD)).isEqualTo("secret"); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.DRIVER)).isEqualTo("postgresql"); } private void checkDatabaseAccess(R2dbcConnectionDetails connectionDetails) { + assertThat(executeQuery(connectionDetails, DatabaseDriver.POSTGRESQL.getValidationQuery(), Integer.class)) + .isEqualTo(1); + } + + private T executeQuery(R2dbcConnectionDetails connectionDetails, String sql, Class resultClass) { ConnectionFactoryOptions connectionFactoryOptions = connectionDetails.getConnectionFactoryOptions(); - Object result = DatabaseClient.create(ConnectionFactories.get(connectionFactoryOptions)) - .sql(DatabaseDriver.POSTGRESQL.getValidationQuery()) - .map((row, metadata) -> row.get(0)) + return DatabaseClient.create(ConnectionFactories.get(connectionFactoryOptions)) + .sql(sql) + .mapValue(resultClass) .first() .block(Duration.ofSeconds(30)); - assertThat(result).isEqualTo(1); } } diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/redis/RedisDockerComposeConnectionDetailsFactoryIntegrationTests.java b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/redis/RedisDockerComposeConnectionDetailsFactoryIntegrationTests.java index 8b6b3d46f5ac..1dec0840d29d 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/redis/RedisDockerComposeConnectionDetailsFactoryIntegrationTests.java +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/redis/RedisDockerComposeConnectionDetailsFactoryIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,9 +16,12 @@ package org.springframework.boot.docker.compose.service.connection.redis; +import javax.net.ssl.SSLContext; + import org.springframework.boot.autoconfigure.data.redis.RedisConnectionDetails; import org.springframework.boot.autoconfigure.data.redis.RedisConnectionDetails.Standalone; import org.springframework.boot.docker.compose.service.connection.test.DockerComposeTest; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testsupport.container.TestImage; import static org.assertj.core.api.Assertions.assertThat; @@ -39,6 +42,17 @@ void runCreatesConnectionDetails(RedisConnectionDetails connectionDetails) { assertConnectionDetails(connectionDetails); } + @DockerComposeTest(composeFile = "redis-ssl-compose.yaml", image = TestImage.REDIS, + additionalResources = { "ca.crt", "server.crt", "server.key", "client.crt", "client.key" }) + void runWithSslCreatesConnectionDetails(RedisConnectionDetails connectionDetails) { + assertConnectionDetails(connectionDetails); + Standalone standalone = connectionDetails.getStandalone(); + SslBundle sslBundle = standalone.getSslBundle(); + assertThat(sslBundle).isNotNull(); + SSLContext sslContext = sslBundle.createSslContext(); + assertThat(sslContext).isNotNull(); + } + @DockerComposeTest(composeFile = "redis-bitnami-compose.yaml", image = TestImage.BITNAMI_REDIS) void runWithBitnamiImageCreatesConnectionDetails(RedisConnectionDetails connectionDetails) { assertConnectionDetails(connectionDetails); diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/test/DockerComposeTest.java b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/test/DockerComposeTest.java index 23a013691d34..ea41f666d7a5 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/test/DockerComposeTest.java +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/test/DockerComposeTest.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -44,6 +44,7 @@ * closed. * * @author Andy Wilkinson + * @author Moritz Halbritter */ @Test @Target(ElementType.METHOD) @@ -63,6 +64,13 @@ */ String composeFile(); + /** + * Additional resources to copy next to the compose file. Loaded as a classpath + * resource relative to the test class. + * @return the additional resources to copy + */ + String[] additionalResources() default {}; + /** * The Docker image reference. * @return the Docker image reference diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/test/DockerComposeTestExtension.java b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/test/DockerComposeTestExtension.java index d2470524b015..e441a7b3215b 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/test/DockerComposeTestExtension.java +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/java/org/springframework/boot/docker/compose/service/connection/test/DockerComposeTestExtension.java @@ -65,6 +65,7 @@ public void beforeTestExecution(ExtensionContext context) throws Exception { store.put(STORE_KEY_WORKSPACE, workspace); try { Path composeFile = prepareComposeFile(workspace, context); + copyAdditionalResources(workspace, context); SpringApplication application = prepareApplication(composeFile); store.put(STORE_KEY_APPLICATION_CONTEXT, application.run()); } @@ -96,6 +97,24 @@ private Path transformedComposeFile(Path workspace, Resource composeFileResource } } + private void copyAdditionalResources(Path workspace, ExtensionContext context) { + DockerComposeTest dockerComposeTest = context.getRequiredTestMethod().getAnnotation(DockerComposeTest.class); + for (String additionalResource : dockerComposeTest.additionalResources()) { + Resource resource = new ClassPathResource(additionalResource, context.getRequiredTestClass()); + copyAdditionalResource(workspace, resource); + } + } + + private void copyAdditionalResource(Path workspace, Resource resource) { + try { + Path source = resource.getFile().toPath(); + Files.copy(source, workspace.resolve(source.getFileName())); + } + catch (IOException ex) { + fail("Error copying additional resource '" + resource + "': " + ex.getMessage(), ex); + } + } + private SpringApplication prepareApplication(Path composeFile) { SpringApplication application = new SpringApplication(Config.class); Map properties = new LinkedHashMap<>(); diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/ldap/lldap-compose.yaml b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/ldap/lldap-compose.yaml new file mode 100644 index 000000000000..c345b52ca2be --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/ldap/lldap-compose.yaml @@ -0,0 +1,8 @@ +services: + ldap: + image: '{imageName}' + environment: + - 'LLDAP_LDAP_BASE_DN=dc=springframework,dc=org' + - 'LLDAP_LDAP_USER_PASS=somepassword' + ports: + - "3890" diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/postgres/postgres-application-name-compose.yaml b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/postgres/postgres-application-name-compose.yaml new file mode 100644 index 000000000000..2b7721344111 --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/postgres/postgres-application-name-compose.yaml @@ -0,0 +1,12 @@ +services: + database: + image: '{imageName}' + ports: + - '5432' + environment: + - 'POSTGRES_USER=myuser' + - 'POSTGRES_DB=mydatabase' + - 'POSTGRES_PASSWORD=secret' + labels: + org.springframework.boot.jdbc.parameters: 'ApplicationName=spring+boot' + org.springframework.boot.r2dbc.parameters: 'applicationName=spring boot' diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/ca.crt b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/ca.crt new file mode 100644 index 000000000000..beed250b132b --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/ca.crt @@ -0,0 +1,32 @@ +-----BEGIN CERTIFICATE----- +MIIFhjCCA26gAwIBAgIUfIkk29IT9OpbgfjL8oRIPSLjUcAwDQYJKoZIhvcNAQEL +BQAwOzEZMBcGA1UECgwQU3ByaW5nIEJvb3QgVGVzdDEeMBwGA1UEAwwVQ2VydGlm +aWNhdGUgQXV0aG9yaXR5MB4XDTI0MDUwMTE2NTMyNVoXDTM0MDQyOTE2NTMyNVow +OzEZMBcGA1UECgwQU3ByaW5nIEJvb3QgVGVzdDEeMBwGA1UEAwwVQ2VydGlmaWNh +dGUgQXV0aG9yaXR5MIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEAusN2 +KzQQUUxZSiI3ZZuZohFwq2KXSUNPdJ6rgD3/YKNTDSZXKZPO53kYPP0DXf0sm3CH +cyWSWVabyimZYuPWena1MElSL4ZpJ9WwkZoOQ3bPFK1utz6kMOwrgAUcky8H/rIK +j2JEBhkSHUIGr57NjUEwG1ygaSerM8RzWw1PtMq+C8LOu3v94qzE3NDg1QRpyvV9 +OmsLsjISd0ZmAJNi9vmiEH923KnPyiqnQmWKpYicdgQmX1GXylS22jZqAwaOkYGj +X8UdeyvrohkZkM0hn9uaSufQGEW4yKACn3PkjJtzi8drBIyjIi9YcAzBxZB9oVKq +XZMlltgO2fDMmIJi0Ngt0Ci7fCoEMqSocKyDKML6YLr9UWtx4bfsrk+rVO9Q/D/v +8RKgstv7dCf2KWRX3ZJEC0IBHS5gLNq0qqqVcGx3LcSyhdiKJOtSwAnNkHMh+jSQ +xLSlBjcSqTPiGTRK/Rddl+xnU/mBgk7ZBGNrUFaD5McMFjddS7Ih82aHnpQ1gekW +nUGv+Tm/G68h2BvZ5U2q+RfeOCgRW9i/AYW2jgT7IFnfjyUXgBQveauMAchomqFE +VLe95ZgViF6vmH34EKo3w9L5TQiwk/r53YlM7TSOTyDqx66t4zGYDsVMicpKmzi4 +2Rp8EpErARRyREUIKSvWs9O9+uT3+7arNLgHe5ECAwEAAaOBgTB/MB0GA1UdDgQW +BBRVMLDVqPECWaH6GruL9E52VcTrPjAfBgNVHSMEGDAWgBRVMLDVqPECWaH6GruL +9E52VcTrPjAPBgNVHRMBAf8EBTADAQH/MCwGA1UdEQQlMCOCC2V4YW1wbGUuY29t +gglsb2NhbGhvc3SCCTEyNy4wLjAuMTANBgkqhkiG9w0BAQsFAAOCAgEAeSpjCL3j +2GIFBNKr/5amLOYa0kZ6r1dJs+K6xvMsUvsBJ/QQsV5nYDMIoV/NYUd8SyYV4lEj +7LHX5ZbmJrvPk30LGEBG/5Vy2MIATrQrQ14S4nXtEdSnBvTQwPOOaHc+2dTp3YpM +f4ffELKWyispTifx1eqdiUJhURKeQBh+3W7zpyaiN4vJaqEDKGgFQtHA/OyZL2hZ +BpxHB0zpb2iDHV8MeyfOT7HQWUk6p13vdYm6EnyJT8fzWvE+TqYNbqFmB+CLRSXy +R3p1yaeTd4LnVknJ0UBKqEyul3ziHZDhKhBpwdglYOQz4eWjSFhikX9XZ8NaI38Q +QqLZVn0DsH2ztkjrQrUVgK2xn4aUuqoLDk4Hu6h5baUn+f2GLuzx+EXc/i3ikYvw +Y3JyufOgw6nGGFG+/QXEj85XtLPhN7Wm42z2e/BGzi0MLl65sfpEDXvFTA72Yzws +OYaeg/HxeYwUHQgs2fKl/LgV4chntSCvTqfNl6OnQafD/ISJNpx3xWR3HwF+ypFG +UaLE+e1soqEJbzL31U/6pypHLsj8Y8r9hJbZXo2ibnhjFV6fypUAP0rbIzaoWcrJ +T0Sbliz+KQTMzCcubiAi4bI/kZ5FJ4kkaHqUpIWzlx1h2WVJ65ASFDjBWb8eVmB6 +Dyno/RVFR/rUL5091gjGRXhLsi1oUHKdEzU= +-----END CERTIFICATE----- diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/client.crt b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/client.crt new file mode 100644 index 000000000000..811d880fcbd3 --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/client.crt @@ -0,0 +1,26 @@ +-----BEGIN CERTIFICATE----- +MIIEWjCCAkKgAwIBAgIURBZvq442tp+/K9TZII5Vy/LzVx0wDQYJKoZIhvcNAQEL +BQAwOzEZMBcGA1UECgwQU3ByaW5nIEJvb3QgVGVzdDEeMBwGA1UEAwwVQ2VydGlm +aWNhdGUgQXV0aG9yaXR5MB4XDTI0MDUwMTE2NTMyNVoXDTM0MDQyOTE2NTMyNVow +LzEZMBcGA1UECgwQU3ByaW5nIEJvb3QgVGVzdDESMBAGA1UEAwwJbG9jYWxob3N0 +MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAvGb7tu0odSuOjeY1lHlh +sRR4PayAvlryjfrrp49hjoVTiL3d/Jo6Po5HlqwJcYuclm0EWQR5Vur/zYJpfUE7 +b8+E9Qwe50+YzfQ2tVFEdq/VfqemrYRGee+pMelOCI90enOKCxfpo6EHbz+WnUP0 +mnD8OAF9QpolSdWAMOGJoPdWX65KQvyMXvQbj9VIHmsx7NCaIOYxjHXB/dI2FmXV ++m4VT6mb8he9dXmgK/ozMq6XIPOAXe0n3dlfMTSEddeNeVwnBpr/n5e0cpwGFhdf +NNu5CI4ecipBhXljJi/4/47M/6hd69HwE05C4zyH4ZDZ2JTfaSKOLV+jYdBUqJP5 +dwIDAQABo2IwYDALBgNVHQ8EBAMCBaAwEQYJYIZIAYb4QgEBBAQDAgeAMB0GA1Ud +DgQWBBRWiWOo9cm2IF/ZlhWLVjifLzYa/DAfBgNVHSMEGDAWgBRVMLDVqPECWaH6 +GruL9E52VcTrPjANBgkqhkiG9w0BAQsFAAOCAgEAA5Wphtu2nBhY+QNOBOwXq4zF +N5qt2IYTLfR7xqpKhhXx9VkIjdPWpcsGuCuMmfPVNvQWE6iK0/jMMqToTj4H6K7e +MN74j0GwwcknT1P42tUzEpg8LKR8VMdhWhyqdniCDNWWuaz1iVSoF0S2i4jFSzH5 +1q3KMKMZ4niK5aJI0fAGa4fCjyuun1Mfg/qGBGwLnqDkIXjeAopZf4Jb64TtzjAs +j9NT6mYbe3E0tw3fHT9ihYdbZDZgSjeCsuq9OiRMVb0DWWmRoLmmOrlN8IJlHV/3 +WyI/ta4Cw5EZ0oaOg0lIyOxXyvElth1xIvh+kdqZSBsU0gNBri6ZIzYbbTh2KTTO +BJHQt9L5naWG27pDrIxBicWXS/MIYonktm3YgCLfuW3kWcVk8bIlNhfcoAYBBgfM +IEYSYEq+bH2IQ+YoWQz3AxjJ8gEuuSUP6R6mYY65FfpjkKgcpGBvw4EIAmqKDtPS +hlLY/F0XVj9KZzrMyH4/vonu+DAb/P7Zmt2fyk/dQO6bAc3ltRmJbJm4VJ2v/T8I +LVu2FtcUYgtLNtkWUPfdb3GSUUgkKlUpWSty31TKSUszJjW1oRykQhEko6o5U3S8 +ptQzXdApsb1lGOqewkubE25tIu2RLiNkKcjFOjJ/lu0vP9k76wWwRVnFLFvfo4lW +pgywiOifs5JbcCt0ZQ0= +-----END CERTIFICATE----- diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/client.key b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/client.key new file mode 100644 index 000000000000..2ae0f49bf4a4 --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/client.key @@ -0,0 +1,28 @@ +-----BEGIN PRIVATE KEY----- +MIIEvgIBADANBgkqhkiG9w0BAQEFAASCBKgwggSkAgEAAoIBAQC8Zvu27Sh1K46N +5jWUeWGxFHg9rIC+WvKN+uunj2GOhVOIvd38mjo+jkeWrAlxi5yWbQRZBHlW6v/N +gml9QTtvz4T1DB7nT5jN9Da1UUR2r9V+p6athEZ576kx6U4Ij3R6c4oLF+mjoQdv +P5adQ/SacPw4AX1CmiVJ1YAw4Ymg91ZfrkpC/Ixe9BuP1UgeazHs0Jog5jGMdcH9 +0jYWZdX6bhVPqZvyF711eaAr+jMyrpcg84Bd7Sfd2V8xNIR11415XCcGmv+fl7Ry +nAYWF18027kIjh5yKkGFeWMmL/j/jsz/qF3r0fATTkLjPIfhkNnYlN9pIo4tX6Nh +0FSok/l3AgMBAAECggEABXnBe3MwXAMQENzNypOiXK4VE3XMYkePfdsSK163byOD +w3ZeTgQNfU4g8LJK8/homzO0SQIJAdz2+ZFbpsp4A2W2zJ+1jvN5RuX/8/UcVhmk +tb1IL/LWCvx5/aoYBWkgIA70UfQJa2jDbdM0v5j/Gu9yE7GI14jh6DFC3xGMGV3b +fOwManxf7sDibCI1nGjnFYNGxninRr+tpb+a1KNbVzhett68LrgPmtph6B3HCPAJ +zBigk1Phgb8WHozTXxnLyw9/RdKJ0Ro4PFmtQv0EvCSlytptnF+0nXkqr3f851XS +bUWwYFchIFWPMhPfD5B3niNWCV42/sU/bQlk+BMQAQKBgQD6NvMq8EdYy2Y7fXT5 +FgB4s+7EkLgI2d5LUaCXCFgc6iZtCTQKUXj1rIWeRfGrFVCCe8qV+XIMKt/G5eEi +tn5ifHhktA2A8GK1scj026qHP3bVn0hMaUnkCF1UpDRKPiEO5G/apPtav8PbCNaX +GAimLGw+WZNZuv7+T33bEBeUdwKBgQDAwiidayLXkRkz2deefdDKcXQsB7RHFGGy +vfZPBCGqizxml+6ojJkkDsVUKL1IXFfyK9KpQAI6tezn4oktgu4jAQqkYY7QZobs +RpQx1dR+KxEm7ISDBTq/B1Q9cFKUKVvQQy8N2pnIbCdzb6MTOKLmJqFGTjr+5T8q +F32B5vkDAQKBgDCKfH42AwFc5EZiPlEcTZcdARMtKCa/bXqbKVZjjgR+AFpi0K+3 +womWoI1l8E5KYkYOEe0qaU+m+aaybgy37qjYkNqoe34qJFwvU1b9ToXScBFdRz9b +pbQRU1naSTKl/u/OrUxzeTfPwAU8H7VMOlFSiOVHp2he+J0JetcGtixdAoGBAIJQ +QMj7rxhxHcqyEVUy1b6nKNTDeJs9Kjd+uU/+CQyVCQaK3GvScY2w9rLIv/51f3dX +LRoDDf7HExxJSFgeVgQQJjOvSK+XQMvngzSVzQxm7TeVWpiBJpAS0l6e2xUTSODp +KpyBFsoqZBlkdaj+9xIFN66iILxGG4fHTbBOiDYBAoGBAOZMKjM5N/hGcCmik/6t +p/zBA2pN9O6zwPndITTsdyVWSlVqCZhXlRX47CerAN+/WVCidlh7Vp5Tuy75Wa77 +v16IDLO01txgWNobcLaM4VgFsyLi5JuxK73S18Vb1cKWdHFRF0LH3cUIq20fjpv6 +Odl4vjNOncXMZCLPHQ+bKWaf +-----END PRIVATE KEY----- diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/redis-ssl-compose.yaml b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/redis-ssl-compose.yaml new file mode 100644 index 000000000000..7f91f4077b46 --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/redis-ssl-compose.yaml @@ -0,0 +1,21 @@ +services: + redis: + image: '{imageName}' + ports: + - '6379' + secrets: + - ssl-ca + - ssl-key + - ssl-cert + command: 'redis-server --tls-port 6379 --port 0 --tls-cert-file /run/secrets/ssl-cert --tls-key-file /run/secrets/ssl-key --tls-ca-cert-file /run/secrets/ssl-ca' + labels: + - 'org.springframework.boot.sslbundle.pem.keystore.certificate=client.crt' + - 'org.springframework.boot.sslbundle.pem.keystore.private-key=client.key' + - 'org.springframework.boot.sslbundle.pem.truststore.certificate=ca.crt' +secrets: + ssl-ca: + file: 'ca.crt' + ssl-key: + file: 'server.key' + ssl-cert: + file: 'server.crt' diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/server.crt b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/server.crt new file mode 100644 index 000000000000..57c66cc78a3b --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/server.crt @@ -0,0 +1,26 @@ +-----BEGIN CERTIFICATE----- +MIIEWjCCAkKgAwIBAgIURBZvq442tp+/K9TZII5Vy/LzVxwwDQYJKoZIhvcNAQEL +BQAwOzEZMBcGA1UECgwQU3ByaW5nIEJvb3QgVGVzdDEeMBwGA1UEAwwVQ2VydGlm +aWNhdGUgQXV0aG9yaXR5MB4XDTI0MDUwMTE2NTMyNVoXDTM0MDQyOTE2NTMyNVow +LzEZMBcGA1UECgwQU3ByaW5nIEJvb3QgVGVzdDESMBAGA1UEAwwJbG9jYWxob3N0 +MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAsllxsSQzTTJlNHMfXC2b +CIXCPsfCgCBl7FbPz828jwJk+EYcXh0+WTFGks0WxSwb8NQza5UtyCUDEueZj9fV +j5mWBY97WCu01Sl/3xClHmYisXfyyv27GKec7PaSOurCm2JDkyHRNumiJROa4jte +N0GOHzw7FYsM3779TuNw14/gtW+eBrGnvgrpU7fbUvx42Di6ftGYQUwIi+3uIaqT +//i7ktDMaAQJtkL6haTzZ5JN2qKO5a34/WRz/ApvPw3lpDV8c4qoTk3C0Bg9MP+a +DnZtjtLBSN9CJWwr+n11QaMgHTotEKsOahGdi3J2zYxCvJP0LT+hjN2O9aRzSMIs +MwIDAQABo2IwYDALBgNVHQ8EBAMCBaAwEQYJYIZIAYb4QgEBBAQDAgZAMB0GA1Ud +DgQWBBS9XQHGwJZhG0olAGM1UMNuwZ65DzAfBgNVHSMEGDAWgBRVMLDVqPECWaH6 +GruL9E52VcTrPjANBgkqhkiG9w0BAQsFAAOCAgEAhBcqm5UQahn8iFMETXvfLMR6 +OOPijsHQ5lVfhig08s46a9O5eaJ9EYSYyiDnxYvZ4gYVH03f/kPwNLamvGR5KIBQ +R0DltkPPX4a11/vjwlSq1cXAt9r59nY+sNcVXWgIWH7zNodL8lyTpYhqvB2wEQkx +t2/JKZ8A0sGjed4S6I5HofYd7bnBxQZgfZShQ2SdDbzbcyg4SCEb8ghwnsH0KNZo +jJF+20RpK2VMViE6lylLTEMd/PyAdST/NPoqVxyva3QjTrKt+tkkFTsmNVMXcmYC +f1xo1/YFp73FFE63VYFI+Yw+Ajau8sYSo4+YvgFCy+Efhf3h3GFDtaiNod56uX9G +9M/cu8XsFzFP2e/0YWY3XL+v7ESOdc3g7yS4FQZ7Z6YvfAed9hCB25cDECvZXqJG +HSYDR38NHyAPROuCwlEwDyVmWRl9bpwZt+hr9kaTQScIDx+rV/EF3o0GKIwtR7AK +jaPAta0f4/Uu+EuWAcccSRUMtfx5/Jse/6iliBvy7JXmA+Y0PrT7K4uHO7iktdI+ +x8WbfZKfnLVuqw5fneTjC1n48Ltjis/f8DgO7BuWTmLdZXddjqqxzBSukFTBn4Hg +/oSg3XiMywOAVrRCNJehcdTG0u/BqZsrRjcYAJaf5qG/0tMLNsuF9Y53XQQAeezE +etL+7y0mkeQhVF+Kmy4= +-----END CERTIFICATE----- diff --git a/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/server.key b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/server.key new file mode 100644 index 000000000000..95e2ef3e8b31 --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/dockerTest/resources/org/springframework/boot/docker/compose/service/connection/redis/server.key @@ -0,0 +1,28 @@ +-----BEGIN PRIVATE KEY----- +MIIEugIBADANBgkqhkiG9w0BAQEFAASCBKQwggSgAgEAAoIBAQCyWXGxJDNNMmU0 +cx9cLZsIhcI+x8KAIGXsVs/PzbyPAmT4RhxeHT5ZMUaSzRbFLBvw1DNrlS3IJQMS +55mP19WPmZYFj3tYK7TVKX/fEKUeZiKxd/LK/bsYp5zs9pI66sKbYkOTIdE26aIl +E5riO143QY4fPDsViwzfvv1O43DXj+C1b54Gsae+CulTt9tS/HjYOLp+0ZhBTAiL +7e4hqpP/+LuS0MxoBAm2QvqFpPNnkk3aoo7lrfj9ZHP8Cm8/DeWkNXxziqhOTcLQ +GD0w/5oOdm2O0sFI30IlbCv6fXVBoyAdOi0Qqw5qEZ2LcnbNjEK8k/QtP6GM3Y71 +pHNIwiwzAgMBAAECgf9REZuCvy2Bi8SoTnjqQuHG5FuA6cPuisuFZr1k88IO+zJQ +uY3WKNs29BV+LcxnoK29W8jQnjqPHXcMfrF5dVWmkrrJdu8JLaGWVHF+uBq8nRb0 +2LvREh5XhZTGzIESNdc/7GIxdouag/8FlzCUYQGuT3v9+wUCiim+4CuIuPvv7ncD +8vANe3Ua5G0mHjVshOiMNpegg45zYlzYpMtUFPs+asLilW6A7UlgC+pLZ1cHUUlU +ZB7KOGT9JdrZpilTidl6LLvDDQK30TSWz8A26SuEAE71DR2VEjLVpjTNS76vlx+c +CrYr/WwpMb0xul+e/uHiNgo+51FiTiJ/IfuGeskCgYEA804CXQM6i5m4/Upps2yG +aTae5xBaYUquZREp5Zb054U6lUAHI41iTMTIwTTvWn5ogNojgi+YjljkzRj2RQ5k +NccBkjBBwwUNVWpBoGeZ73KAdejNB4C4ucGc2kkqEDo4MU5x3IE4JK1Yi1jl9mKb +IR6m3pqb2PCQHjO8sqKNHYkCgYEAu6fH/qUd/XGmCZJWY5K6jg3dISXH16MTO5M+ +jetprkGMMybWKZQa1GedXurPexE48oRlRhkjdQkW6Wcj1Qh6OKp6N2Zx8sY4dLeQ +yVChnMPFE2LK+UlRCKJUZi+rzX415ML6pZg+yW7O2cHpMKv7PlXISw2YDqtboCAi +Y+doqNsCgYBE1yqmBJbZDuqfiCF2KduyA0lcmWzpIEdNw1h2ZIrwwup7dj1O2t8Y +V4lx2TdsBF4vLwli+XKRvCcovMpZaaQC70bLhSnmMxS9uS3OY+HTNTORqQfx+oLJ +1DU8Mf1b0A08LjTbLhijkASAkOuoFehMq66NR3OXIyGz2fGnHYUN+QKBgCC47SL2 +X/hl7PIWVoIef/FtcXXqRKLRiPUGhA3zUwZT38K7rvSpItSPDN4UTAHFywxfEdnb +YFd0Mk6Y8aKgS8+9ynoGnzAaaJXRvKmeKdBQQvlSbNpzcnHy/IylG2xF6dfuOA7Q +MYKmk+Nc8PDPzIveIYMU58MHFn8hm12YaKOpAoGAV1CE8hFkEK9sbRGoKNJkx9nm +CZTv7PybaG/RN4ZrBSwVmnER0FEagA/Tzrlp1pi3sC8ZsC9onSOf6Btq8ZE0zbO1 +vsAm3gTBXcrCJxzw0Wjt8pzEbk3yELm4WE6VDEx4da2jWocdspslpIwdjHnPwsbH +r5O3ZAgigZs/ZtKW/U4= +-----END PRIVATE KEY----- diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultConnectionPorts.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultConnectionPorts.java index 2981385ecb66..1b7999845c95 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultConnectionPorts.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultConnectionPorts.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -97,7 +97,8 @@ private Map buildMappingsForHostNetworking(Config config @Override public int get(int containerPort) { Integer hostPort = this.portMappings.get(containerPort); - Assert.state(hostPort != null, "No host port mapping found for container port %s".formatted(containerPort)); + Assert.state(hostPort != null, + () -> "No host port mapping found for container port %s".formatted(containerPort)); return hostPort; } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultDockerCompose.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultDockerCompose.java index e50340605b14..612412b67796 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultDockerCompose.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultDockerCompose.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -102,7 +102,7 @@ public List getRunningServices() { Map inspected = inspect(runningPsResponses); for (DockerCliComposePsResponse psResponse : runningPsResponses) { DockerCliInspectResponse inspectResponse = inspectContainer(psResponse.id(), inspected); - Assert.notNull(inspectResponse, () -> "Failed to inspect container '%s'".formatted(psResponse.id())); + Assert.state(inspectResponse != null, () -> "Failed to inspect container '%s'".formatted(psResponse.id())); result.add(new DefaultRunningService(this.hostname, dockerComposeFile, psResponse, inspectResponse)); } return Collections.unmodifiableList(result); diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultRunningService.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultRunningService.java index 700ec8e76aba..2509d82c81df 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultRunningService.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DefaultRunningService.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,6 +45,8 @@ class DefaultRunningService implements RunningService, OriginProvider { private final DockerEnv env; + private final DockerComposeFile composeFile; + DefaultRunningService(DockerHost host, DockerComposeFile composeFile, DockerCliComposePsResponse composePsResponse, DockerCliInspectResponse inspectResponse) { this.origin = new DockerComposeOrigin(composeFile, composePsResponse.name()); @@ -55,6 +57,7 @@ class DefaultRunningService implements RunningService, OriginProvider { this.ports = new DefaultConnectionPorts(inspectResponse); this.env = new DockerEnv(inspectResponse.config().env()); this.labels = Collections.unmodifiableMap(inspectResponse.config().labels()); + this.composeFile = composeFile; } @Override @@ -97,4 +100,9 @@ public String toString() { return this.name; } + @Override + public DockerComposeFile composeFile() { + return this.composeFile; + } + } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DockerComposeFile.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DockerComposeFile.java index 30156ed4c46b..41c099cfb110 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DockerComposeFile.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/DockerComposeFile.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ public final class DockerComposeFile { private final List files; private DockerComposeFile(List files) { - Assert.state(!files.isEmpty(), "Files must not be empty"); + Assert.isTrue(!files.isEmpty(), "'files' must not be empty"); this.files = files.stream().map(DockerComposeFile::toCanonicalFile).toList(); } @@ -112,7 +112,7 @@ public static DockerComposeFile find(File workingDirectory) { if (!base.exists()) { return null; } - Assert.isTrue(base.isDirectory(), () -> "'%s' is not a directory".formatted(base)); + Assert.state(base.isDirectory(), () -> "'%s' is not a directory".formatted(base)); Path basePath = base.toPath(); for (String candidate : SEARCH_ORDER) { Path resolved = basePath.resolve(candidate); @@ -129,9 +129,9 @@ public static DockerComposeFile find(File workingDirectory) { * @return the Docker Compose file */ public static DockerComposeFile of(File file) { - Assert.notNull(file, "File must not be null"); - Assert.isTrue(file.exists(), () -> "Docker Compose file '%s' does not exist".formatted(file)); - Assert.isTrue(file.isFile(), () -> "Docker compose file '%s' is not a file".formatted(file)); + Assert.notNull(file, "'file' must not be null"); + Assert.isTrue(file.exists(), () -> "'file' [%s] must exist".formatted(file)); + Assert.isTrue(file.isFile(), () -> "'file' [%s] must be a normal file".formatted(file)); return new DockerComposeFile(Collections.singletonList(file)); } @@ -142,11 +142,11 @@ public static DockerComposeFile of(File file) { * @since 3.4.0 */ public static DockerComposeFile of(Collection files) { - Assert.notNull(files, "Files must not be null"); + Assert.notNull(files, "'files' must not be null"); for (File file : files) { - Assert.notNull(file, "File must not be null"); - Assert.isTrue(file.exists(), () -> "Docker Compose file '%s' does not exist".formatted(file)); - Assert.isTrue(file.isFile(), () -> "Docker compose file '%s' is not a file".formatted(file)); + Assert.notNull(file, "'files' must not contain null elements"); + Assert.isTrue(file.exists(), () -> "'files' content [%s] must exist".formatted(file)); + Assert.isTrue(file.isFile(), () -> "'files' content [%s] must be a normal file".formatted(file)); } return new DockerComposeFile(List.copyOf(files)); } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/ImageName.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/ImageName.java index 931ce4d2272c..5f0aee3b31d1 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/ImageName.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/ImageName.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,7 @@ class ImageName { private final String string; ImageName(String domain, String path) { - Assert.hasText(path, "Path must not be empty"); + Assert.hasText(path, "'path' must not be empty"); this.domain = getDomainOrDefault(domain); this.name = getNameWithDefaultPath(this.domain, path); this.string = this.domain + "/" + this.name; @@ -114,13 +114,12 @@ static String parseDomain(String value) { } static ImageName of(String value) { - Assert.hasText(value, "Value must not be empty"); + Assert.hasText(value, "'value' must not be empty"); String domain = parseDomain(value); String path = (domain != null) ? value.substring(domain.length() + 1) : value; Assert.isTrue(Regex.PATH.matcher(path).matches(), - () -> "Unable to parse name \"" + value + "\". " - + "Image name must be in the form '[domainHost:port/][path/]name', " - + "with 'path' and 'name' containing only [a-z0-9][.][_][-]"); + () -> "'value' path must contain an image reference in the form '[domainHost:port/][path/]name' " + + "(with 'path' and 'name' containing only [a-z0-9][.][_][-]) [" + value + "]"); return new ImageName(domain, path); } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/ImageReference.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/ImageReference.java index 33c0ba873102..32d1eba4458e 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/ImageReference.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/ImageReference.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,7 @@ public final class ImageReference { private final String string; private ImageReference(ImageName name, String tag, String digest) { - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); this.name = name; this.tag = tag; this.digest = digest; @@ -136,7 +136,7 @@ private String buildString(String name, String tag, String digest) { * @return an {@link ImageReference} instance */ public static ImageReference of(String value) { - Assert.hasText(value, "Value must not be null"); + Assert.hasText(value, "'value' must not be null"); String domain = ImageName.parseDomain(value); String path = (domain != null) ? value.substring(domain.length() + 1) : value; String digest = null; @@ -162,9 +162,9 @@ public static ImageReference of(String value) { } } Assert.isTrue(Regex.PATH.matcher(path).matches(), - () -> "Unable to parse image reference \"" + value + "\". " - + "Image reference must be in the form '[domainHost:port/][path/]name[:tag][@digest]', " - + "with 'path' and 'name' containing only [a-z0-9][.][_][-]"); + () -> "'value' path must contain an image reference in the form " + + "'[domainHost:port/][path/]name[:tag][@digest] " + + "(with 'path' and 'name' containing only [a-z0-9][.][_][-]) [" + value + "]"); ImageName name = new ImageName(domain, path); return new ImageReference(name, tag, digest); } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/RunningService.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/RunningService.java index eefc8caffe7d..80e0ee0b854b 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/RunningService.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/core/RunningService.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -64,4 +64,13 @@ public interface RunningService { */ Map labels(); + /** + * Return the Docker Compose file for the service. + * @return the Docker Compose file + * @since 3.5.0 + */ + default DockerComposeFile composeFile() { + return null; + } + } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/ConnectionNamePredicate.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/ConnectionNamePredicate.java index 1990f9b27134..3a32f79a3882 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/ConnectionNamePredicate.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/ConnectionNamePredicate.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ class ConnectionNamePredicate implements Predicate required; ConnectionNamePredicate(String... required) { - Assert.notEmpty(required, "Required must not be empty"); + Assert.notEmpty(required, "'required' must not be empty"); this.required = Arrays.stream(required).map(this::asCanonicalName).collect(Collectors.toSet()); } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeConnectionDetailsFactory.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeConnectionDetailsFactory.java index 302a3ba35817..7fae0976290b 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,17 +16,33 @@ package org.springframework.boot.docker.compose.service.connection; +import java.nio.file.Path; import java.util.Arrays; +import java.util.Set; import java.util.function.Predicate; import org.springframework.boot.autoconfigure.service.connection.ConnectionDetails; import org.springframework.boot.autoconfigure.service.connection.ConnectionDetailsFactory; +import org.springframework.boot.docker.compose.core.DockerComposeFile; import org.springframework.boot.docker.compose.core.RunningService; +import org.springframework.boot.io.ApplicationResourceLoader; import org.springframework.boot.origin.Origin; import org.springframework.boot.origin.OriginProvider; +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslBundleKey; +import org.springframework.boot.ssl.SslOptions; +import org.springframework.boot.ssl.jks.JksSslStoreBundle; +import org.springframework.boot.ssl.jks.JksSslStoreDetails; +import org.springframework.boot.ssl.pem.PemSslStore; +import org.springframework.boot.ssl.pem.PemSslStoreBundle; +import org.springframework.boot.ssl.pem.PemSslStoreDetails; +import org.springframework.core.io.ResourceLoader; +import org.springframework.core.io.support.SpringFactoriesLoader; import org.springframework.util.Assert; import org.springframework.util.ClassUtils; +import org.springframework.util.CollectionUtils; import org.springframework.util.ObjectUtils; +import org.springframework.util.StringUtils; /** * Base class for {@link ConnectionDetailsFactory} implementations that provide @@ -106,12 +122,14 @@ protected static class DockerComposeConnectionDetails implements ConnectionDetai private final Origin origin; + private volatile SslBundle sslBundle; + /** * Create a new {@link DockerComposeConnectionDetails} instance. * @param runningService the source {@link RunningService} */ protected DockerComposeConnectionDetails(RunningService runningService) { - Assert.notNull(runningService, "RunningService must not be null"); + Assert.notNull(runningService, "'runningService' must not be null"); this.origin = Origin.from(runningService); } @@ -120,6 +138,113 @@ public Origin getOrigin() { return this.origin; } + protected SslBundle getSslBundle(RunningService service) { + if (this.sslBundle != null) { + return this.sslBundle; + } + SslBundle jksSslBundle = getJksSslBundle(service); + SslBundle pemSslBundle = getPemSslBundle(service); + if (jksSslBundle == null && pemSslBundle == null) { + return null; + } + if (jksSslBundle != null && pemSslBundle != null) { + throw new IllegalStateException("Mutually exclusive JKS and PEM ssl bundles have been configured"); + } + SslBundle sslBundle = (jksSslBundle != null) ? jksSslBundle : pemSslBundle; + this.sslBundle = sslBundle; + return sslBundle; + } + + private SslBundle getJksSslBundle(RunningService service) { + JksSslStoreDetails keyStoreDetails = getJksSslStoreDetails(service, "keystore"); + JksSslStoreDetails trustStoreDetails = getJksSslStoreDetails(service, "truststore"); + if (keyStoreDetails == null && trustStoreDetails == null) { + return null; + } + SslBundleKey key = SslBundleKey.of(service.labels().get("org.springframework.boot.sslbundle.jks.key.alias"), + service.labels().get("org.springframework.boot.sslbundle.jks.key.password")); + SslOptions options = createSslOptions( + service.labels().get("org.springframework.boot.sslbundle.jks.options.ciphers"), + service.labels().get("org.springframework.boot.sslbundle.jks.options.enabled-protocols")); + String protocol = service.labels().get("org.springframework.boot.sslbundle.jks.protocol"); + Path workingDirectory = getWorkingDirectory(service); + return SslBundle.of( + new JksSslStoreBundle(keyStoreDetails, trustStoreDetails, getResourceLoader(workingDirectory)), key, + options, protocol); + } + + private ResourceLoader getResourceLoader(Path workingDirectory) { + ClassLoader classLoader = ApplicationResourceLoader.get().getClassLoader(); + return ApplicationResourceLoader.get(classLoader, + SpringFactoriesLoader.forDefaultResourceLocation(classLoader), workingDirectory); + } + + private JksSslStoreDetails getJksSslStoreDetails(RunningService service, String storeType) { + String type = service.labels().get("org.springframework.boot.sslbundle.jks.%s.type".formatted(storeType)); + String provider = service.labels() + .get("org.springframework.boot.sslbundle.jks.%s.provider".formatted(storeType)); + String location = service.labels() + .get("org.springframework.boot.sslbundle.jks.%s.location".formatted(storeType)); + String password = service.labels() + .get("org.springframework.boot.sslbundle.jks.%s.password".formatted(storeType)); + if (location == null) { + return null; + } + return new JksSslStoreDetails(type, provider, location, password); + } + + private Path getWorkingDirectory(RunningService runningService) { + DockerComposeFile composeFile = runningService.composeFile(); + if (composeFile == null || CollectionUtils.isEmpty(composeFile.getFiles())) { + return Path.of("."); + } + return composeFile.getFiles().get(0).toPath().getParent(); + } + + private SslOptions createSslOptions(String ciphers, String enabledProtocols) { + Set ciphersSet = null; + if (StringUtils.hasLength(ciphers)) { + ciphersSet = StringUtils.commaDelimitedListToSet(ciphers); + } + Set enabledProtocolsSet = null; + if (StringUtils.hasLength(enabledProtocols)) { + enabledProtocolsSet = StringUtils.commaDelimitedListToSet(enabledProtocols); + } + return SslOptions.of(ciphersSet, enabledProtocolsSet); + } + + private SslBundle getPemSslBundle(RunningService service) { + PemSslStoreDetails keyStoreDetails = getPemSslStoreDetails(service, "keystore"); + PemSslStoreDetails trustStoreDetails = getPemSslStoreDetails(service, "truststore"); + if (keyStoreDetails == null && trustStoreDetails == null) { + return null; + } + SslBundleKey key = SslBundleKey.of(service.labels().get("org.springframework.boot.sslbundle.pem.key.alias"), + service.labels().get("org.springframework.boot.sslbundle.pem.key.password")); + SslOptions options = createSslOptions( + service.labels().get("org.springframework.boot.sslbundle.pem.options.ciphers"), + service.labels().get("org.springframework.boot.sslbundle.pem.options.enabled-protocols")); + String protocol = service.labels().get("org.springframework.boot.sslbundle.pem.protocol"); + Path workingDirectory = getWorkingDirectory(service); + ResourceLoader resourceLoader = getResourceLoader(workingDirectory); + return SslBundle.of(new PemSslStoreBundle(PemSslStore.load(keyStoreDetails, resourceLoader), + PemSslStore.load(trustStoreDetails, resourceLoader)), key, options, protocol); + } + + private PemSslStoreDetails getPemSslStoreDetails(RunningService service, String storeType) { + String type = service.labels().get("org.springframework.boot.sslbundle.pem.%s.type".formatted(storeType)); + String certificate = service.labels() + .get("org.springframework.boot.sslbundle.pem.%s.certificate".formatted(storeType)); + String privateKey = service.labels() + .get("org.springframework.boot.sslbundle.pem.%s.private-key".formatted(storeType)); + String privateKeyPassword = service.labels() + .get("org.springframework.boot.sslbundle.pem.%s.private-key-password".formatted(storeType)); + if (certificate == null && privateKey == null) { + return null; + } + return new PemSslStoreDetails(type, certificate, privateKey, privateKeyPassword); + } + } } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeConnectionSource.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeConnectionSource.java index 4ddcc877612c..f75b8dd5ab3e 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeConnectionSource.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeConnectionSource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,6 +17,7 @@ package org.springframework.boot.docker.compose.service.connection; import org.springframework.boot.docker.compose.core.RunningService; +import org.springframework.core.env.Environment; /** * Passed to {@link DockerComposeConnectionDetailsFactory} to provide details of the @@ -32,12 +33,16 @@ public final class DockerComposeConnectionSource { private final RunningService runningService; + private final Environment environment; + /** * Create a new {@link DockerComposeConnectionSource} instance. * @param runningService the running Docker Compose service + * @param environment environment in which the current application is running */ - DockerComposeConnectionSource(RunningService runningService) { + DockerComposeConnectionSource(RunningService runningService, Environment environment) { this.runningService = runningService; + this.environment = environment; } /** @@ -48,4 +53,13 @@ public RunningService getRunningService() { return this.runningService; } + /** + * Environment in which the current application is running. + * @return the environment + * @since 3.5.0 + */ + public Environment getEnvironment() { + return this.environment; + } + } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeServiceConnectionsApplicationListener.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeServiceConnectionsApplicationListener.java index 4cf5135d26ed..61acfcf99052 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeServiceConnectionsApplicationListener.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/DockerComposeServiceConnectionsApplicationListener.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,6 +31,7 @@ import org.springframework.boot.docker.compose.lifecycle.DockerComposeServicesReadyEvent; import org.springframework.context.ApplicationContext; import org.springframework.context.ApplicationListener; +import org.springframework.core.env.Environment; import org.springframework.util.ClassUtils; import org.springframework.util.StringUtils; @@ -59,13 +60,15 @@ class DockerComposeServiceConnectionsApplicationListener public void onApplicationEvent(DockerComposeServicesReadyEvent event) { ApplicationContext applicationContext = event.getSource(); if (applicationContext instanceof BeanDefinitionRegistry registry) { - registerConnectionDetails(registry, event.getRunningServices()); + Environment environment = applicationContext.getEnvironment(); + registerConnectionDetails(registry, environment, event.getRunningServices()); } } - private void registerConnectionDetails(BeanDefinitionRegistry registry, List runningServices) { + private void registerConnectionDetails(BeanDefinitionRegistry registry, Environment environment, + List runningServices) { for (RunningService runningService : runningServices) { - DockerComposeConnectionSource source = new DockerComposeConnectionSource(runningService); + DockerComposeConnectionSource source = new DockerComposeConnectionSource(runningService, environment); this.factories.getConnectionDetails(source, false).forEach((connectionDetailsType, connectionDetails) -> { register(registry, runningService, connectionDetailsType, connectionDetails); this.factories.getConnectionDetails(connectionDetails, false) diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/jdbc/JdbcUrlBuilder.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/jdbc/JdbcUrlBuilder.java index da4815943c4a..078ace52e966 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/jdbc/JdbcUrlBuilder.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/jdbc/JdbcUrlBuilder.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,7 +42,7 @@ public class JdbcUrlBuilder { * @param containerPort the source container port */ public JdbcUrlBuilder(String driverProtocol, int containerPort) { - Assert.notNull(driverProtocol, "DriverProtocol must not be null"); + Assert.notNull(driverProtocol, "'driverProtocol' must not be null"); this.driverProtocol = driverProtocol; this.containerPort = containerPort; } @@ -67,7 +67,7 @@ public String build(RunningService service, String database) { } private String urlFor(RunningService service, String database) { - Assert.notNull(service, "Service must not be null"); + Assert.notNull(service, "'service' must not be null"); StringBuilder url = new StringBuilder("jdbc:%s://%s:%d".formatted(this.driverProtocol, service.host(), service.ports().get(this.containerPort))); if (StringUtils.hasLength(database)) { diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/ldap/LLdapDockerComposeConnectionDetailsFactory.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/ldap/LLdapDockerComposeConnectionDetailsFactory.java new file mode 100644 index 000000000000..0c72c224adfd --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/ldap/LLdapDockerComposeConnectionDetailsFactory.java @@ -0,0 +1,92 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.docker.compose.service.connection.ldap; + +import java.util.Map; + +import org.springframework.boot.autoconfigure.ldap.LdapConnectionDetails; +import org.springframework.boot.docker.compose.core.RunningService; +import org.springframework.boot.docker.compose.service.connection.DockerComposeConnectionDetailsFactory; +import org.springframework.boot.docker.compose.service.connection.DockerComposeConnectionSource; + +/** + * {@link DockerComposeConnectionDetailsFactory} to create {@link LdapConnectionDetails} + * for an {@code ldap} service. + * + * @author Eddú Meléndez + */ +class LLdapDockerComposeConnectionDetailsFactory extends DockerComposeConnectionDetailsFactory { + + LLdapDockerComposeConnectionDetailsFactory() { + super("lldap/lldap"); + } + + @Override + protected LdapConnectionDetails getDockerComposeConnectionDetails(DockerComposeConnectionSource source) { + return new LLdapDockerComposeConnectionDetails(source.getRunningService()); + } + + /** + * {@link LdapConnectionDetails} backed by an {@code lldap} {@link RunningService}. + */ + static class LLdapDockerComposeConnectionDetails extends DockerComposeConnectionDetails + implements LdapConnectionDetails { + + private final String[] urls; + + private final String base; + + private final String username; + + private final String password; + + LLdapDockerComposeConnectionDetails(RunningService service) { + super(service); + Map env = service.env(); + boolean usesTls = Boolean.parseBoolean(env.getOrDefault("LLDAP_LDAPS_OPTIONS__ENABLED", "false")); + String ldapPort = usesTls ? env.getOrDefault("LLDAP_LDAPS_OPTIONS__PORT", "6360") + : env.getOrDefault("LLDAP_LDAP_PORT", "3890"); + this.urls = new String[] { "%s://%s:%d".formatted(usesTls ? "ldaps" : "ldap", service.host(), + service.ports().get(Integer.parseInt(ldapPort))) }; + this.base = env.getOrDefault("LLDAP_LDAP_BASE_DN", "dc=example,dc=com"); + this.password = env.getOrDefault("LLDAP_LDAP_USER_PASS", "password"); + this.username = "cn=admin,ou=people,%s".formatted(this.base); + } + + @Override + public String[] getUrls() { + return this.urls; + } + + @Override + public String getBase() { + return this.base; + } + + @Override + public String getUsername() { + return this.username; + } + + @Override + public String getPassword() { + return this.password; + } + + } + +} diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresEnvironment.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresEnvironment.java index 220dbbf3e55b..a4f64eae03a1 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresEnvironment.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresEnvironment.java @@ -67,8 +67,9 @@ private String extractPassword(Map env) { return null; } String password = env.getOrDefault("POSTGRES_PASSWORD", env.get("POSTGRESQL_PASSWORD")); - Assert.state(StringUtils.hasLength(password), "PostgreSQL password must be provided"); - return password; + boolean allowEmpty = env.containsKey("ALLOW_EMPTY_PASSWORD"); + Assert.state(allowEmpty || StringUtils.hasLength(password), "No PostgreSQL password found"); + return (password != null) ? password : ""; } private boolean isUsingTrustHostAuthMethod(Map env) { diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactory.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactory.java index 3f4ad8653d26..ec82a16ac941 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,11 +16,16 @@ package org.springframework.boot.docker.compose.service.connection.postgres; +import java.net.URLEncoder; +import java.nio.charset.StandardCharsets; + import org.springframework.boot.autoconfigure.jdbc.JdbcConnectionDetails; import org.springframework.boot.docker.compose.core.RunningService; import org.springframework.boot.docker.compose.service.connection.DockerComposeConnectionDetailsFactory; import org.springframework.boot.docker.compose.service.connection.DockerComposeConnectionSource; import org.springframework.boot.docker.compose.service.connection.jdbc.JdbcUrlBuilder; +import org.springframework.core.env.Environment; +import org.springframework.util.StringUtils; /** * {@link DockerComposeConnectionDetailsFactory} to create {@link JdbcConnectionDetails} @@ -42,7 +47,7 @@ protected PostgresJdbcDockerComposeConnectionDetailsFactory() { @Override protected JdbcConnectionDetails getDockerComposeConnectionDetails(DockerComposeConnectionSource source) { - return new PostgresJdbcDockerComposeConnectionDetails(source.getRunningService()); + return new PostgresJdbcDockerComposeConnectionDetails(source.getRunningService(), source.getEnvironment()); } /** @@ -57,10 +62,11 @@ static class PostgresJdbcDockerComposeConnectionDetails extends DockerComposeCon private final String jdbcUrl; - PostgresJdbcDockerComposeConnectionDetails(RunningService service) { + PostgresJdbcDockerComposeConnectionDetails(RunningService service, Environment environment) { super(service); this.environment = new PostgresEnvironment(service.env()); - this.jdbcUrl = jdbcUrlBuilder.build(service, this.environment.getDatabase()); + this.jdbcUrl = addApplicationNameIfNecessary(jdbcUrlBuilder.build(service, this.environment.getDatabase()), + environment); } @Override @@ -78,6 +84,27 @@ public String getJdbcUrl() { return this.jdbcUrl; } + private static String addApplicationNameIfNecessary(String jdbcUrl, Environment environment) { + if (jdbcUrl.contains("&ApplicationName=") || jdbcUrl.contains("?ApplicationName=")) { + return jdbcUrl; + } + String applicationName = environment.getProperty("spring.application.name"); + if (!StringUtils.hasText(applicationName)) { + return jdbcUrl; + } + StringBuilder jdbcUrlBuilder = new StringBuilder(jdbcUrl); + if (!jdbcUrl.contains("?")) { + jdbcUrlBuilder.append("?"); + } + else if (!jdbcUrl.endsWith("&")) { + jdbcUrlBuilder.append("&"); + } + return jdbcUrlBuilder.append("ApplicationName") + .append('=') + .append(URLEncoder.encode(applicationName, StandardCharsets.UTF_8)) + .toString(); + } + } } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactory.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactory.java index cb1c66ded50b..6ac8ea6d7c1d 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,12 +17,15 @@ package org.springframework.boot.docker.compose.service.connection.postgres; import io.r2dbc.spi.ConnectionFactoryOptions; +import io.r2dbc.spi.Option; import org.springframework.boot.autoconfigure.r2dbc.R2dbcConnectionDetails; import org.springframework.boot.docker.compose.core.RunningService; import org.springframework.boot.docker.compose.service.connection.DockerComposeConnectionDetailsFactory; import org.springframework.boot.docker.compose.service.connection.DockerComposeConnectionSource; import org.springframework.boot.docker.compose.service.connection.r2dbc.ConnectionFactoryOptionsBuilder; +import org.springframework.core.env.Environment; +import org.springframework.util.StringUtils; /** * {@link DockerComposeConnectionDetailsFactory} to create {@link R2dbcConnectionDetails} @@ -44,7 +47,7 @@ class PostgresR2dbcDockerComposeConnectionDetailsFactory @Override protected R2dbcConnectionDetails getDockerComposeConnectionDetails(DockerComposeConnectionSource source) { - return new PostgresDbR2dbcDockerComposeConnectionDetails(source.getRunningService()); + return new PostgresDbR2dbcDockerComposeConnectionDetails(source.getRunningService(), source.getEnvironment()); } /** @@ -53,16 +56,16 @@ protected R2dbcConnectionDetails getDockerComposeConnectionDetails(DockerCompose static class PostgresDbR2dbcDockerComposeConnectionDetails extends DockerComposeConnectionDetails implements R2dbcConnectionDetails { + private static final Option APPLICATION_NAME = Option.valueOf("applicationName"); + private static final ConnectionFactoryOptionsBuilder connectionFactoryOptionsBuilder = new ConnectionFactoryOptionsBuilder( "postgresql", 5432); private final ConnectionFactoryOptions connectionFactoryOptions; - PostgresDbR2dbcDockerComposeConnectionDetails(RunningService service) { + PostgresDbR2dbcDockerComposeConnectionDetails(RunningService service, Environment environment) { super(service); - PostgresEnvironment environment = new PostgresEnvironment(service.env()); - this.connectionFactoryOptions = connectionFactoryOptionsBuilder.build(service, environment.getDatabase(), - environment.getUsername(), environment.getPassword()); + this.connectionFactoryOptions = getConnectionFactoryOptions(service, environment); } @Override @@ -70,6 +73,26 @@ public ConnectionFactoryOptions getConnectionFactoryOptions() { return this.connectionFactoryOptions; } + private static ConnectionFactoryOptions getConnectionFactoryOptions(RunningService service, + Environment environment) { + PostgresEnvironment env = new PostgresEnvironment(service.env()); + ConnectionFactoryOptions connectionFactoryOptions = connectionFactoryOptionsBuilder.build(service, + env.getDatabase(), env.getUsername(), env.getPassword()); + return addApplicationNameIfNecessary(connectionFactoryOptions, environment); + } + + private static ConnectionFactoryOptions addApplicationNameIfNecessary( + ConnectionFactoryOptions connectionFactoryOptions, Environment environment) { + if (connectionFactoryOptions.hasOption(APPLICATION_NAME)) { + return connectionFactoryOptions; + } + String applicationName = environment.getProperty("spring.application.name"); + if (!StringUtils.hasText(applicationName)) { + return connectionFactoryOptions; + } + return connectionFactoryOptions.mutate().option(APPLICATION_NAME, applicationName).build(); + } + } } diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/r2dbc/ConnectionFactoryOptionsBuilder.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/r2dbc/ConnectionFactoryOptionsBuilder.java index 3157443079eb..42c0ded490a5 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/r2dbc/ConnectionFactoryOptionsBuilder.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/r2dbc/ConnectionFactoryOptionsBuilder.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -51,14 +51,14 @@ public class ConnectionFactoryOptionsBuilder { * @param containerPort the source container port */ public ConnectionFactoryOptionsBuilder(String driver, int containerPort) { - Assert.notNull(driver, "Driver must not be null"); + Assert.notNull(driver, "'driver' must not be null"); this.driver = driver; this.sourcePort = containerPort; } public ConnectionFactoryOptions build(RunningService service, String database, String user, String password) { - Assert.notNull(service, "Service must not be null"); - Assert.notNull(database, "Database must not be null"); + Assert.notNull(service, "'service' must not be null"); + Assert.notNull(database, "'database' must not be null"); ConnectionFactoryOptions.Builder builder = ConnectionFactoryOptions.builder() .option(ConnectionFactoryOptions.DRIVER, this.driver) .option(ConnectionFactoryOptions.HOST, service.host()) @@ -91,7 +91,7 @@ private Map parseParameters(String parameters) { Map result = new LinkedHashMap<>(); for (String parameter : StringUtils.commaDelimitedListToStringArray(parameters)) { String[] parts = parameter.split("="); - Assert.state(parts.length == 2, () -> "Unable to parse parameter '%s'".formatted(parameter)); + Assert.state(parts.length == 2, () -> "'parameters' [%s] must cotain parsable value".formatted(parameter)); result.put(parts[0], parts[1]); } return Collections.unmodifiableMap(result); diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/redis/RedisDockerComposeConnectionDetailsFactory.java b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/redis/RedisDockerComposeConnectionDetailsFactory.java index 5a15429d2f94..bc3f47dc37c6 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/redis/RedisDockerComposeConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-docker-compose/src/main/java/org/springframework/boot/docker/compose/service/connection/redis/RedisDockerComposeConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -57,7 +57,7 @@ static class RedisDockerComposeConnectionDetails extends DockerComposeConnection RedisDockerComposeConnectionDetails(RunningService service) { super(service); - this.standalone = Standalone.of(service.host(), service.ports().get(REDIS_PORT)); + this.standalone = Standalone.of(service.host(), service.ports().get(REDIS_PORT), getSslBundle(service)); } @Override diff --git a/spring-boot-project/spring-boot-docker-compose/src/main/resources/META-INF/spring.factories b/spring-boot-project/spring-boot-docker-compose/src/main/resources/META-INF/spring.factories index ad1a5dfc2416..d2defa268464 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/main/resources/META-INF/spring.factories +++ b/spring-boot-project/spring-boot-docker-compose/src/main/resources/META-INF/spring.factories @@ -14,6 +14,7 @@ org.springframework.boot.docker.compose.service.connection.clickhouse.ClickHouse org.springframework.boot.docker.compose.service.connection.elasticsearch.ElasticsearchDockerComposeConnectionDetailsFactory,\ org.springframework.boot.docker.compose.service.connection.flyway.JdbcAdaptingFlywayConnectionDetailsFactory,\ org.springframework.boot.docker.compose.service.connection.hazelcast.HazelcastDockerComposeConnectionDetailsFactory,\ +org.springframework.boot.docker.compose.service.connection.ldap.LLdapDockerComposeConnectionDetailsFactory,\ org.springframework.boot.docker.compose.service.connection.ldap.OpenLdapDockerComposeConnectionDetailsFactory,\ org.springframework.boot.docker.compose.service.connection.liquibase.JdbcAdaptingLiquibaseConnectionDetailsFactory,\ org.springframework.boot.docker.compose.service.connection.mariadb.MariaDbJdbcDockerComposeConnectionDetailsFactory,\ diff --git a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/DockerComposeFileTests.java b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/DockerComposeFileTests.java index 95d3892d69e2..114dce29cd6b 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/DockerComposeFileTests.java +++ b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/DockerComposeFileTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,6 +27,7 @@ import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; /** * Tests for {@link DockerComposeFile}. @@ -104,7 +105,7 @@ void findWhenWorkingDirectoryDoesNotExistReturnsNull() { @Test void findWhenWorkingDirectoryIsNotDirectoryThrowsException() throws Exception { File file = createTempFile("iamafile"); - assertThatIllegalArgumentException().isThrownBy(() -> DockerComposeFile.find(file)) + assertThatIllegalStateException().isThrownBy(() -> DockerComposeFile.find(file)) .withMessageEndingWith("is not a directory"); } @@ -128,20 +129,20 @@ void ofWithMultipleFilesReturnsDockerComposeFile() throws Exception { @Test void ofWhenFileIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> DockerComposeFile.of((File) null)) - .withMessage("File must not be null"); + .withMessage("'file' must not be null"); } @Test void ofWhenFileDoesNotExistThrowsException() { File file = new File(this.temp, "missing"); assertThatIllegalArgumentException().isThrownBy(() -> DockerComposeFile.of(file)) - .withMessageEndingWith("does not exist"); + .withMessageEndingWith("must exist"); } @Test void ofWhenFileIsNotFileThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> DockerComposeFile.of(this.temp)) - .withMessageEndingWith("is not a file"); + .withMessageEndingWith("must be a normal file"); } private DockerComposeFile createComposeFile(String name) throws IOException { diff --git a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/ImageNameTests.java b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/ImageNameTests.java index 76f7fb3228e3..ea40164be052 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/ImageNameTests.java +++ b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/ImageNameTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -118,25 +118,26 @@ void ofWhenLegacyDomainUsesNewDomain() { @Test void ofWhenNameIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of(null)) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test void ofWhenNameIsEmptyThrowsException() { - assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of("")).withMessage("Value must not be empty"); + assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of("")) + .withMessage("'value' must not be empty"); } @Test void ofWhenContainsUppercaseThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of("Test")) - .withMessageContaining("Unable to parse name") + .withMessageContaining("must contain an image reference") .withMessageContaining("Test"); } @Test void ofWhenNameIncludesTagThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of("ubuntu:latest")) - .withMessageContaining("Unable to parse name") + .withMessageContaining("must contain an image reference") .withMessageContaining(":latest"); } @@ -144,7 +145,7 @@ void ofWhenNameIncludesTagThrowsException() { void ofWhenNameIncludeDigestThrowsException() { assertThatIllegalArgumentException().isThrownBy( () -> ImageName.of("ubuntu@sha256:47bfdb88c3ae13e488167607973b7688f69d9e8c142c2045af343ec199649c09")) - .withMessageContaining("Unable to parse name") + .withMessageContaining("must contain an image reference") .withMessageContaining("@sha256:47b"); } diff --git a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/ImageReferenceTests.java b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/ImageReferenceTests.java index e9eb9045819f..caff7a7092b2 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/ImageReferenceTests.java +++ b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/core/ImageReferenceTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -165,7 +165,7 @@ void ofWhenHasIllegalCharacterThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> ImageReference .of("registry.example.com/example/example-app:1.6.0-dev.2.uncommitted+wip.foo.c75795d")) - .withMessageContaining("Unable to parse image reference"); + .withMessageContaining("must contain an image reference"); } @Test @@ -173,7 +173,7 @@ void ofWhenHasIllegalCharacterThrowsException() { void ofWhenImageNameIsVeryLongAndHasIllegalCharacterThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ImageReference .of("docker.io/library/this-image-has-a-long-name-with-an-invalid-tag-which-is-at-danger-of-catastrophic-backtracking:1.0.0+1234")) - .withMessageContaining("Unable to parse image reference"); + .withMessageContaining("must contain an image reference"); } @Test diff --git a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/jdbc/JdbcUrlBuilderTests.java b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/jdbc/JdbcUrlBuilderTests.java index bb0589a16ae1..f2cc0cc09c86 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/jdbc/JdbcUrlBuilderTests.java +++ b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/jdbc/JdbcUrlBuilderTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,7 +43,7 @@ class JdbcUrlBuilderTests { @Test void createWhenDriverProtocolIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new JdbcUrlBuilder(null, 123)) - .withMessage("DriverProtocol must not be null"); + .withMessage("'driverProtocol' must not be null"); } @Test @@ -84,7 +84,7 @@ protected void appendParameters(StringBuilder url, String parameters) { @Test void buildWhenServiceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.builder.build(null, "mydb")) - .withMessage("Service must not be null"); + .withMessage("'service' must not be null"); } private RunningService mockService(int mappedPort) { diff --git a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresEnvironmentTests.java b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresEnvironmentTests.java index 24ce50bc11df..83cec657c1cc 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresEnvironmentTests.java +++ b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresEnvironmentTests.java @@ -39,7 +39,7 @@ class PostgresEnvironmentTests { @Test void createWhenNoPostgresPasswordThrowsException() { assertThatIllegalStateException().isThrownBy(() -> new PostgresEnvironment(Collections.emptyMap())) - .withMessage("PostgreSQL password must be provided"); + .withMessage("No PostgreSQL password found"); } @Test @@ -93,6 +93,12 @@ void getPasswordWhenHasTrustHostAuthMethod() { assertThat(environment.getPassword()).isNull(); } + @Test + void getPasswordWhenHasNoPasswordAndAllowEmptyPassword() { + PostgresEnvironment environment = new PostgresEnvironment(Map.of("ALLOW_EMPTY_PASSWORD", "yes")); + assertThat(environment.getPassword()).isEmpty(); + } + @Test void getDatabaseWhenNoPostgresDbOrPostgresUser() { PostgresEnvironment environment = new PostgresEnvironment(Map.of("POSTGRES_PASSWORD", "secret")); diff --git a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests.java b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests.java new file mode 100644 index 000000000000..0795351868b4 --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresJdbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests.java @@ -0,0 +1,120 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.docker.compose.service.connection.postgres; + +import java.util.LinkedHashMap; +import java.util.Map; + +import org.junit.jupiter.api.Test; + +import org.springframework.boot.autoconfigure.jdbc.JdbcConnectionDetails; +import org.springframework.boot.docker.compose.core.ConnectionPorts; +import org.springframework.boot.docker.compose.core.RunningService; +import org.springframework.mock.env.MockEnvironment; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.BDDMockito.given; +import static org.mockito.Mockito.mock; + +/** + * Tests for + * {@link PostgresJdbcDockerComposeConnectionDetailsFactory.PostgresJdbcDockerComposeConnectionDetails}. + * + * @author Dmytro Nosan + */ +class PostgresJdbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests { + + private final RunningService service = mock(RunningService.class); + + private final MockEnvironment environment = new MockEnvironment(); + + private final Map labels = new LinkedHashMap<>(); + + PostgresJdbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests() { + given(this.service.env()) + .willReturn(Map.of("POSTGRES_USER", "user", "POSTGRES_PASSWORD", "password", "POSTGRES_DB", "database")); + given(this.service.labels()).willReturn(this.labels); + ConnectionPorts connectionPorts = mock(ConnectionPorts.class); + given(this.service.ports()).willReturn(connectionPorts); + given(this.service.host()).willReturn("localhost"); + given(connectionPorts.get(5432)).willReturn(30001); + } + + @Test + void createConnectionDetails() { + JdbcConnectionDetails connectionDetails = getConnectionDetails(); + assertConnectionDetails(connectionDetails); + assertThat(connectionDetails.getJdbcUrl()).endsWith("/database"); + } + + @Test + void createConnectionDetailsWithLabels() { + this.labels.put("org.springframework.boot.jdbc.parameters", "connectTimeout=30&ApplicationName=spring-boot"); + JdbcConnectionDetails connectionDetails = getConnectionDetails(); + assertConnectionDetails(connectionDetails); + assertThat(connectionDetails.getJdbcUrl()).endsWith("?connectTimeout=30&ApplicationName=spring-boot"); + } + + @Test + void createConnectionDetailsWithApplicationNameLabelTakesPrecedence() { + this.labels.put("org.springframework.boot.jdbc.parameters", "ApplicationName=spring-boot"); + this.environment.setProperty("spring.application.name", "my-app"); + JdbcConnectionDetails connectionDetails = getConnectionDetails(); + assertConnectionDetails(connectionDetails); + assertThat(connectionDetails.getJdbcUrl()).endsWith("?ApplicationName=spring-boot"); + } + + @Test + void createConnectionDetailsWithSpringApplicationName() { + this.environment.setProperty("spring.application.name", "spring boot"); + JdbcConnectionDetails connectionDetails = getConnectionDetails(); + assertConnectionDetails(connectionDetails); + assertThat(connectionDetails.getJdbcUrl()).endsWith("?ApplicationName=spring+boot"); + } + + @Test + void createConnectionDetailsAppendSpringApplicationName() { + this.labels.put("org.springframework.boot.jdbc.parameters", "connectTimeout=30"); + this.environment.setProperty("spring.application.name", "spring boot"); + JdbcConnectionDetails connectionDetails = getConnectionDetails(); + assertConnectionDetails(connectionDetails); + assertThat(connectionDetails.getJdbcUrl()).endsWith("?connectTimeout=30&ApplicationName=spring+boot"); + } + + @Test + void createConnectionDetailsAppendSpringApplicationNameParametersEndedWithAmpersand() { + this.labels.put("org.springframework.boot.jdbc.parameters", "connectTimeout=30&"); + this.environment.setProperty("spring.application.name", "spring boot"); + JdbcConnectionDetails connectionDetails = getConnectionDetails(); + assertConnectionDetails(connectionDetails); + assertThat(connectionDetails.getJdbcUrl()).endsWith("?connectTimeout=30&ApplicationName=spring+boot"); + + } + + private void assertConnectionDetails(JdbcConnectionDetails connectionDetails) { + assertThat(connectionDetails.getUsername()).isEqualTo("user"); + assertThat(connectionDetails.getPassword()).isEqualTo("password"); + assertThat(connectionDetails.getJdbcUrl()).startsWith("jdbc:postgresql://localhost:30001/database"); + assertThat(connectionDetails.getDriverClassName()).isEqualTo("org.postgresql.Driver"); + } + + private JdbcConnectionDetails getConnectionDetails() { + return new PostgresJdbcDockerComposeConnectionDetailsFactory.PostgresJdbcDockerComposeConnectionDetails( + this.service, this.environment); + } + +} diff --git a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests.java b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests.java new file mode 100644 index 000000000000..a39774867965 --- /dev/null +++ b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/postgres/PostgresR2dbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests.java @@ -0,0 +1,119 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.docker.compose.service.connection.postgres; + +import java.util.LinkedHashMap; +import java.util.Map; + +import io.r2dbc.spi.ConnectionFactoryOptions; +import io.r2dbc.spi.Option; +import org.junit.jupiter.api.Test; + +import org.springframework.boot.docker.compose.core.ConnectionPorts; +import org.springframework.boot.docker.compose.core.RunningService; +import org.springframework.mock.env.MockEnvironment; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.BDDMockito.given; +import static org.mockito.Mockito.mock; + +/** + * Tests for + * {@link PostgresR2dbcDockerComposeConnectionDetailsFactory.PostgresDbR2dbcDockerComposeConnectionDetails}. + * + * @author Dmytro Nosan + */ +class PostgresR2dbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests { + + private static final Option APPLICATION_NAME = Option.valueOf("applicationName"); + + private final RunningService service = mock(RunningService.class); + + private final MockEnvironment environment = new MockEnvironment(); + + private final Map labels = new LinkedHashMap<>(); + + PostgresR2dbcDockerComposeConnectionDetailsFactoryConnectionDetailsTests() { + given(this.service.env()) + .willReturn(Map.of("POSTGRES_USER", "myuser", "POSTGRES_PASSWORD", "secret", "POSTGRES_DB", "mydatabase")); + given(this.service.labels()).willReturn(this.labels); + ConnectionPorts connectionPorts = mock(ConnectionPorts.class); + given(this.service.ports()).willReturn(connectionPorts); + given(this.service.host()).willReturn("localhost"); + given(connectionPorts.get(5432)).willReturn(30001); + } + + @Test + void createConnectionDetails() { + ConnectionFactoryOptions options = getConnectionFactoryOptions(); + assertConnectionFactoryOptions(options); + assertThat(options.getValue(APPLICATION_NAME)).isNull(); + } + + @Test + void createConnectionDetailsWithLabels() { + this.labels.put("org.springframework.boot.r2dbc.parameters", + "connectTimeout=PT15S,applicationName=spring-boot"); + ConnectionFactoryOptions options = getConnectionFactoryOptions(); + assertConnectionFactoryOptions(options); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.CONNECT_TIMEOUT)).isEqualTo("PT15S"); + assertThat(options.getRequiredValue(APPLICATION_NAME)).isEqualTo("spring-boot"); + } + + @Test + void createConnectionDetailsWithApplicationNameLabelTakesPrecedence() { + this.labels.put("org.springframework.boot.r2dbc.parameters", "applicationName=spring-boot"); + this.environment.setProperty("spring.application.name", "my-app"); + ConnectionFactoryOptions options = getConnectionFactoryOptions(); + assertConnectionFactoryOptions(options); + assertThat(options.getRequiredValue(APPLICATION_NAME)).isEqualTo("spring-boot"); + } + + @Test + void createConnectionDetailsWithSpringApplicationName() { + this.environment.setProperty("spring.application.name", "spring boot"); + ConnectionFactoryOptions options = getConnectionFactoryOptions(); + assertConnectionFactoryOptions(options); + assertThat(options.getRequiredValue(APPLICATION_NAME)).isEqualTo("spring boot"); + } + + @Test + void createConnectionDetailsAppendSpringApplicationName() { + this.labels.put("org.springframework.boot.r2dbc.parameters", "connectTimeout=PT15S"); + this.environment.setProperty("spring.application.name", "my-app"); + ConnectionFactoryOptions options = getConnectionFactoryOptions(); + assertConnectionFactoryOptions(options); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.CONNECT_TIMEOUT)).isEqualTo("PT15S"); + assertThat(options.getRequiredValue(APPLICATION_NAME)).isEqualTo("my-app"); + } + + private void assertConnectionFactoryOptions(ConnectionFactoryOptions options) { + assertThat(options.getRequiredValue(ConnectionFactoryOptions.HOST)).isEqualTo("localhost"); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.PORT)).isEqualTo(30001); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.DATABASE)).isEqualTo("mydatabase"); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.USER)).isEqualTo("myuser"); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.PASSWORD)).isEqualTo("secret"); + assertThat(options.getRequiredValue(ConnectionFactoryOptions.DRIVER)).isEqualTo("postgresql"); + } + + private ConnectionFactoryOptions getConnectionFactoryOptions() { + return new PostgresR2dbcDockerComposeConnectionDetailsFactory.PostgresDbR2dbcDockerComposeConnectionDetails( + this.service, this.environment) + .getConnectionFactoryOptions(); + } + +} diff --git a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/r2dbc/ConnectionFactoryOptionsBuilderTests.java b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/r2dbc/ConnectionFactoryOptionsBuilderTests.java index 91fd383c3687..0e75c4f90f30 100644 --- a/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/r2dbc/ConnectionFactoryOptionsBuilderTests.java +++ b/spring-boot-project/spring-boot-docker-compose/src/test/java/org/springframework/boot/docker/compose/service/connection/r2dbc/ConnectionFactoryOptionsBuilderTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -46,7 +46,7 @@ class ConnectionFactoryOptionsBuilderTests { @Test void createWhenDriverProtocolIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new JdbcUrlBuilder(null, 123)) - .withMessage("DriverProtocol must not be null"); + .withMessage("'driverProtocol' must not be null"); } @Test @@ -81,14 +81,14 @@ void buildWhenHasParamsLabelBuildsOptions() { @Test void buildWhenServiceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.builder.build(null, "mydb", "user", "pass")) - .withMessage("Service must not be null"); + .withMessage("'service' must not be null"); } @Test void buildWhenDatabaseIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> this.builder.build(mockService(456), null, "user", "pass")) - .withMessage("Database must not be null"); + .withMessage("'database' must not be null"); } private RunningService mockService(int mappedPort) { diff --git a/spring-boot-project/spring-boot-docs/build.gradle b/spring-boot-project/spring-boot-docs/build.gradle index ba25b57d84be..87a99024488b 100644 --- a/spring-boot-project/spring-boot-docs/build.gradle +++ b/spring-boot-project/spring-boot-docs/build.gradle @@ -1,3 +1,5 @@ +import org.springframework.boot.build.docs.ConfigureJavadocLinks + plugins { id "dev.adamko.dokkatoo-html" id "java" @@ -14,6 +16,7 @@ configurations { autoConfiguration configurationProperties remoteSpringApplicationExample + resolvedBom springApplicationExample testSlices all { @@ -78,6 +81,7 @@ dependencies { implementation(project(path: ":spring-boot-project:spring-boot-tools:spring-boot-cli")) implementation(project(path: ":spring-boot-project:spring-boot-tools:spring-boot-loader-tools")) implementation("ch.qos.logback:logback-classic") + implementation("com.redis:testcontainers-redis") implementation("com.zaxxer:HikariCP") implementation("io.micrometer:micrometer-jakarta9") implementation("io.micrometer:micrometer-tracing") @@ -170,6 +174,7 @@ dependencies { implementation("org.testcontainers:junit-jupiter") implementation("org.testcontainers:neo4j") implementation("org.testcontainers:mongodb") + implementation("org.testcontainers:elasticsearch") implementation("org.junit.jupiter:junit-jupiter") implementation("org.yaml:snakeyaml") @@ -178,6 +183,8 @@ dependencies { remoteSpringApplicationExample(project(":spring-boot-project:spring-boot-starters:spring-boot-starter-logging")) remoteSpringApplicationExample("org.springframework:spring-web") + resolvedBom(project(path: ":spring-boot-project:spring-boot-dependencies", configuration: "resolvedBom")) + springApplicationExample(platform(project(":spring-boot-project:spring-boot-dependencies"))) springApplicationExample(project(path: ":spring-boot-project:spring-boot-starters:spring-boot-starter-web")) @@ -198,7 +205,6 @@ dokkatoo { } task aggregatedJavadoc(type: Javadoc) { - dependsOn dependencyVersions project.rootProject.gradle.projectsEvaluated { Set publishedProjects = rootProject.subprojects.findAll { it != project } .findAll { it.plugins.hasPlugin(JavaPlugin) && it.plugins.hasPlugin(MavenPublishPlugin) } @@ -220,24 +226,7 @@ task aggregatedJavadoc(type: Javadoc) { use = true windowTitle = "Spring Boot ${project.version} API" } - doFirst { - def versionConstraints = dependencyVersions.versionConstraints - def toMajorMinorVersion = version -> { - String formatted = version.split("\\.").take(2).join('.') + '.x' - return version.endsWith("-SNAPSHOT") ? formatted + "-SNAPSHOT" : formatted - } - def springFrameworkVersion = toMajorMinorVersion(versionConstraints["org.springframework:spring-core"]) - def springSecurityVersion = toMajorMinorVersion(versionConstraints["org.springframework.security:spring-security-core"]) - def tomcatVersion = "${versionConstraints["org.apache.tomcat:tomcat-annotations-api"]}" - def tomcatDocsVersion = tomcatVersion.substring(0, tomcatVersion.lastIndexOf(".")); - options.links = [ - "https://docs.oracle.com/en/java/javase/17/docs/api/", - "https://docs.spring.io/spring-framework/docs/${springFrameworkVersion}/javadoc-api/", - "https://docs.spring.io/spring-security/site/docs/${springSecurityVersion}/api/", - "https://jakarta.ee/specifications/platform/9/apidocs/", - "https://tomcat.apache.org/tomcat-${tomcatDocsVersion}-doc/api/", - ] as String[] - } + doFirst(new ConfigureJavadocLinks(configurations.resolvedBom, ["Spring Framework", "Spring Security", "Tomcat"])) } } @@ -255,16 +244,14 @@ task documentAutoConfigurationClasses(type: org.springframework.boot.build.autoc outputDir = layout.buildDirectory.dir("generated/docs/auto-configuration-classes/documented-auto-configuration-classes/") } -task documentDependencyVersionCoordinates(type: org.springframework.boot.build.constraints.DocumentConstrainedVersions) { - dependsOn dependencyVersions - constrainedVersions.set(providers.provider { dependencyVersions.constrainedVersions }) +task documentDependencyVersionCoordinates(type: org.springframework.boot.build.docs.DocumentManagedDependencies) { outputFile = layout.buildDirectory.file("generated/docs/dependency-versions/documented-coordinates.adoc") + resolvedBoms = configurations.resolvedBom } -task documentDependencyVersionProperties(type: org.springframework.boot.build.constraints.DocumentVersionProperties) { - dependsOn dependencyVersions - versionProperties.set(providers.provider { dependencyVersions.versionProperties}) +task documentDependencyVersionProperties(type: org.springframework.boot.build.docs.DocumentVersionProperties) { outputFile = layout.buildDirectory.file("generated/docs/dependency-versions/documented-properties.adoc") + resolvedBoms = configurations.resolvedBom } task documentConfigurationProperties(type: org.springframework.boot.build.context.properties.DocumentConfigurationProperties) { diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/ROOT/pages/redirect.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/ROOT/pages/redirect.adoc index ec4127f1ae33..55c60bdaf9df 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/ROOT/pages/redirect.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/ROOT/pages/redirect.adoc @@ -1134,8 +1134,6 @@ * xref:reference:actuator/metrics.adoc#actuator.metrics.export.otlp[#actuator.metrics.export.otlp] * xref:reference:actuator/metrics.adoc#actuator.metrics.export.prometheus[#actuator.metrics.export.prometheus] * xref:reference:actuator/metrics.adoc#actuator.metrics.export.prometheus[#production-ready-metrics-export-prometheus] -* xref:reference:actuator/metrics.adoc#actuator.metrics.export.signalfx[#actuator.metrics.export.signalfx] -* xref:reference:actuator/metrics.adoc#actuator.metrics.export.signalfx[#production-ready-metrics-export-signalfx] * xref:reference:actuator/metrics.adoc#actuator.metrics.export.simple[#actuator.metrics.export.simple] * xref:reference:actuator/metrics.adoc#actuator.metrics.export.simple[#production-ready-metrics-export-simple] * xref:reference:actuator/metrics.adoc#actuator.metrics.export.stackdriver[#actuator.metrics.export.stackdriver] diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/build.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/build.adoc index 5372c5bfb00b..863600502661 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/build.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/build.adoc @@ -112,7 +112,7 @@ Gradle users can achieve the same result by using the {url-cyclonedx-docs-gradle [source,gradle] ---- plugins { - id 'org.cyclonedx.bom' version '1.10.0' + id 'org.cyclonedx.bom' version '2.2.0' } ---- diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/native-image/testing-native-applications.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/native-image/testing-native-applications.adoc index 1ccefe199d34..aafe2c7c38f2 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/native-image/testing-native-applications.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/native-image/testing-native-applications.adoc @@ -85,7 +85,9 @@ You should have a `` section that looks like this: ---- -The `spring-boot-starter-parent` declares a `nativeTest` profile that configures the executions that are needed to run the native tests. +The `spring-boot-starter-parent` defines a `nativeTest` profile that provides the necessary configuration for the Spring Boot and Native Build Tools plugins. +First you need to add those two plugin in the module to opt-in for the feature. +Your tests are executed in native mode only when the `nativeTest` is enabled. You can activate profiles using the `-P` flag on the command line. TIP: If you don't want to use `spring-boot-starter-parent` you'll need to configure executions for the `process-test-aot` goal from the Spring Boot plugin and the `test` goal from the Native Build Tools plugin. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/spring-mvc.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/spring-mvc.adoc index 9defe3fab787..d5278a153696 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/spring-mvc.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/spring-mvc.adoc @@ -118,6 +118,8 @@ Such customizer beans can be ordered (Boot's own customizer has an order of 0), Any beans of type javadoc:com.fasterxml.jackson.databind.Module[] are automatically registered with the auto-configured javadoc:org.springframework.http.converter.json.Jackson2ObjectMapperBuilder[] and are applied to any javadoc:com.fasterxml.jackson.databind.ObjectMapper[] instances that it creates. This provides a global mechanism for contributing custom modules when you add new features to your application. +NOTE: If you wish to register additional modules programmatically using a javadoc:org.springframework.boot.autoconfigure.jackson.Jackson2ObjectMapperBuilderCustomizer[], make sure to use the `modulesToInstall` method that takes a consumer as the other variants are not additive. + If you want to replace the default javadoc:com.fasterxml.jackson.databind.ObjectMapper[] completely, either define a javadoc:org.springframework.context.annotation.Bean[format=annotation] of that type or, if you prefer the builder-based approach, define a javadoc:org.springframework.http.converter.json.Jackson2ObjectMapperBuilder[] javadoc:org.springframework.context.annotation.Bean[format=annotation]. When defining an javadoc:com.fasterxml.jackson.databind.ObjectMapper[] bean, marking it as javadoc:org.springframework.context.annotation.Primary[format=annotation] is recommended as the auto-configuration's javadoc:com.fasterxml.jackson.databind.ObjectMapper[] that it will replace is javadoc:org.springframework.context.annotation.Primary[format=annotation]. Note that, in either case, doing so disables all auto-configuration of the javadoc:com.fasterxml.jackson.databind.ObjectMapper[]. @@ -178,6 +180,7 @@ spring: ---- If you have additional servlets you can declare a javadoc:org.springframework.context.annotation.Bean[format=annotation] of type javadoc:jakarta.servlet.Servlet[] or javadoc:org.springframework.boot.web.servlet.ServletRegistrationBean[] for each and Spring Boot will register them transparently to the container. +It is also possible to use javadoc:org.springframework.boot.web.servlet.ServletRegistration[format=annotation] as an annotation-based alternative to javadoc:org.springframework.boot.web.servlet.ServletRegistrationBean[]. Because servlets are registered that way, they can be mapped to a sub-context of the javadoc:org.springframework.web.servlet.DispatcherServlet[] without invoking it. Configuring the javadoc:org.springframework.web.servlet.DispatcherServlet[] yourself is unusual but if you really need to do it, a javadoc:org.springframework.context.annotation.Bean[format=annotation] of type javadoc:org.springframework.boot.autoconfigure.web.servlet.DispatcherServletPath[] must be provided as well to provide the path of your custom javadoc:org.springframework.web.servlet.DispatcherServlet[]. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/webserver.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/webserver.adoc index fff0b690400e..b51b7606a56f 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/webserver.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/how-to/pages/webserver.adoc @@ -369,6 +369,7 @@ However, you must be very careful that they do not cause eager initialization of You can work around such restrictions by initializing the beans lazily when first used instead of on initialization. In the case of filters and servlets, you can also add mappings and init parameters by adding a javadoc:org.springframework.boot.web.servlet.FilterRegistrationBean[] or a javadoc:org.springframework.boot.web.servlet.ServletRegistrationBean[] instead of or in addition to the underlying component. +You can also use javadoc:org.springframework.boot.web.servlet.ServletRegistration[format=annotation] and javadoc:org.springframework.boot.web.servlet.FilterRegistration[format=annotation] as an annotation-based alternative to javadoc:org.springframework.boot.web.servlet.ServletRegistrationBean[] and javadoc:org.springframework.boot.web.servlet.FilterRegistrationBean[]. [NOTE] ==== diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/metrics.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/metrics.adoc index f3e1d0d4f6bf..fb32b1ad1f92 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/metrics.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/metrics.adoc @@ -17,7 +17,6 @@ Spring Boot Actuator provides dependency management and auto-configuration for { - xref:actuator/metrics.adoc#actuator.metrics.export.newrelic[] - xref:actuator/metrics.adoc#actuator.metrics.export.otlp[] - xref:actuator/metrics.adoc#actuator.metrics.export.prometheus[] -- xref:actuator/metrics.adoc#actuator.metrics.export.signalfx[] - xref:actuator/metrics.adoc#actuator.metrics.export.simple[] (in-memory) - xref:actuator/metrics.adoc#actuator.metrics.export.stackdriver[] - xref:actuator/metrics.adoc#actuator.metrics.export.statsd[] @@ -552,16 +551,13 @@ Please check the https://prometheus.io/docs/prometheus/latest/feature_flags/#exe For ephemeral or batch jobs that may not exist long enough to be scraped, you can use https://github.com/prometheus/pushgateway[Prometheus Pushgateway] support to expose the metrics to Prometheus. -NOTE: The Prometheus Pushgateway only works with the deprecated Prometheus simpleclient for now, until the Prometheus 1.x client adds support for it. -To switch to the simpleclient, remove `io.micrometer:micrometer-registry-prometheus` from your project and add `io.micrometer:micrometer-registry-prometheus-simpleclient` instead. - To enable Prometheus Pushgateway support, add the following dependency to your project: [source,xml] ---- io.prometheus - simpleclient_pushgateway + io.prometheus:prometheus-metrics-exporter-pushgateway ---- @@ -573,34 +569,6 @@ For advanced configuration, you can also provide your own javadoc:org.springfram -[[actuator.metrics.export.signalfx]] -=== SignalFx - -SignalFx registry periodically pushes metrics to {url-micrometer-docs-implementations}/signalFx[SignalFx]. -To export metrics to https://www.signalfx.com[SignalFx], you must provide your access token: - -[configprops,yaml] ----- -management: - signalfx: - metrics: - export: - access-token: "YOUR_ACCESS_TOKEN" ----- - -You can also change the interval at which metrics are sent to SignalFx: - -[configprops,yaml] ----- -management: - signalfx: - metrics: - export: - step: "30s" ----- - - - [[actuator.metrics.export.simple]] === Simple @@ -736,6 +704,7 @@ The following JVM metrics are provided: * Various memory and buffer pool details * Statistics related to garbage collection * Thread utilization +* https://docs.micrometer.io/micrometer/reference/reference/jvm.html#_java_21_metrics[Virtual threads statistics] (for this, `io.micrometer:micrometer-java21` has to be on the classpath) * The number of classes loaded and unloaded * JVM version information * JIT compilation time @@ -806,7 +775,6 @@ See the {url-spring-framework-docs}/integration/observability.html#observability To add to the default tags, provide a javadoc:org.springframework.context.annotation.Bean[format=annotation] that extends javadoc:org.springframework.http.server.observation.DefaultServerRequestObservationConvention[] from the `org.springframework.http.server.observation` package. To replace the default tags, provide a javadoc:org.springframework.context.annotation.Bean[format=annotation] that implements javadoc:org.springframework.http.server.observation.ServerRequestObservationConvention[]. - TIP: In some cases, exceptions handled in web controllers are not recorded as request metrics tags. Applications can opt in and record exceptions by xref:web/servlet.adoc#web.servlet.spring-mvc.error-handling[setting handled exceptions as request attributes]. @@ -865,6 +833,33 @@ To customize the tags, provide a javadoc:org.springframework.context.annotation. +[[actuator.metrics.supported.ssl]] +=== SSL bundle metrics + +Spring Boot Actuator publishes two metrics about SSL bundles: + +The metric `ssl.chains` gauges how many certificate chains have been registered. +The `status` tag can be used to differentiate between valid, not-yet-valid, expired and soon-to-be-expired certificates. + +The metric `ssl.chain.expiry` gauges the expiry date of each certificate chain in seconds. +This number will be negative if the chain has already expired. +This metric is tagged with the following information: + +|=== +| Tag | Description + +| `bundle` +| The name of the bundle which contains the certificate chain + +| `certificate` +| The serial number (in hex format) of the certificate which is the soonest to expire in the chain + +| `chain` +| The name of the certificate chain. +|=== + + + [[actuator.metrics.supported.http-clients]] === HTTP Client Metrics diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/observability.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/observability.adoc index 349ae05782ec..f74fff89fc2f 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/observability.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/observability.adoc @@ -96,11 +96,19 @@ Spring Boot's actuator module includes basic support for OpenTelemetry. It provides a bean of type javadoc:io.opentelemetry.api.OpenTelemetry[], and if there are beans of type javadoc:io.opentelemetry.sdk.trace.SdkTracerProvider[], javadoc:io.opentelemetry.context.propagation.ContextPropagators[], javadoc:io.opentelemetry.sdk.logs.SdkLoggerProvider[] or javadoc:io.opentelemetry.sdk.metrics.SdkMeterProvider[] in the application context, they automatically get registered. Additionally, it provides a javadoc:io.opentelemetry.sdk.resources.Resource[] bean. The attributes of the auto-configured javadoc:io.opentelemetry.sdk.resources.Resource[] can be configured via the configprop:management.opentelemetry.resource-attributes[] configuration property. +Auto-configured attributes will be merged with attributes from the `OTEL_RESOURCE_ATTRIBUTES` and `OTEL_SERVICE_NAME` environment variables, with attributes configured through the configuration property taking precedence over those from the environment variables. + + If you have defined your own javadoc:io.opentelemetry.sdk.resources.Resource[] bean, this will no longer be the case. NOTE: Spring Boot does not provide auto-configuration for OpenTelemetry metrics or logging. OpenTelemetry tracing is only auto-configured when used together with xref:actuator/tracing.adoc[Micrometer Tracing]. +NOTE: The `OTEL_RESOURCE_ATTRIBUTES` environment variable consists of a list of key-value pairs. +For example: `key1=value1,key2=value2,key3=spring%20boot`. +All attribute values are treated as strings, and any characters outside the baggage-octet range must be **percent-encoded**. + + The next sections will provide more details about logging, metrics and traces. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/tracing.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/tracing.adoc index c66a07beabeb..42bae1e728d2 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/tracing.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/actuator/tracing.adoc @@ -150,6 +150,10 @@ Tracing with OpenTelemetry and reporting using OTLP requires the following depen Use the `management.otlp.tracing.*` configuration properties to configure reporting using OTLP. +NOTE: If you need to apply advanced customizations to OTLP span exporters, consider registering javadoc:org.springframework.boot.actuate.autoconfigure.tracing.otlp.OtlpHttpSpanExporterBuilderCustomizer[] or javadoc:org.springframework.boot.actuate.autoconfigure.tracing.otlp.OtlpGrpcSpanExporterBuilderCustomizer[] beans. +These will be invoked before the creation of the `OtlpHttpSpanExporter` or `OtlpGrpcSpanExporter`. +The customizers take precedence over anything applied by the auto-configuration. + [[actuator.micrometer-tracing.tracer-implementations.brave-zipkin]] diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/data/nosql.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/data/nosql.adoc index 245dac77f773..3710fcf6c3be 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/data/nosql.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/data/nosql.adoc @@ -440,7 +440,7 @@ There is a `spring-boot-starter-data-cassandra` starter for collecting the depen [[data.nosql.cassandra.connecting]] === Connecting to Cassandra -You can inject an auto-configured javadoc:org.springframework.data.cassandra.core.CassandraTemplate[] or a Cassandra `CqlSession` instance as you would with any other Spring Bean. +You can inject an auto-configured javadoc:org.springframework.data.cassandra.core.cql.CqlTemplate[], javadoc:org.springframework.data.cassandra.core.CassandraTemplate[], or a Cassandra `CqlSession` instance as you would with any other Spring Bean. The `spring.cassandra.*` properties can be used to customize the connection. Generally, you provide `keyspace-name` and `contact-points` as well the local datacenter name, as shown in the following example: diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/data/sql.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/data/sql.adoc index 3d7a538e1142..4a1615f10f22 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/data/sql.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/data/sql.adoc @@ -142,6 +142,7 @@ The following connection pools are supported by javadoc:org.springframework.boot * H2 javadoc:org.h2.jdbcx.JdbcDataSource[] * PostgreSQL javadoc:org.postgresql.ds.PGSimpleDataSource[] * C3P0 +* Vibur @@ -179,6 +180,8 @@ spring: max-rows: 500 ---- +If tuning of SQL exceptions is required, you can define your own `SQLExceptionTranslator` bean so that it is associated with the auto-configured `JdbcTemplate`. + NOTE: The javadoc:org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate[] reuses the same javadoc:org.springframework.jdbc.core.JdbcTemplate[] instance behind the scenes. If more than one javadoc:org.springframework.jdbc.core.JdbcTemplate[] is defined and no primary candidate exists, the javadoc:org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate[] is not auto-configured. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/dev-services.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/dev-services.adoc index ed9f44407336..af495d1af2b4 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/dev-services.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/dev-services.adoc @@ -102,7 +102,7 @@ The following service connections are currently supported: | Containers named "clickhouse/clickhouse-server", "bitnami/clickhouse", "gvenzl/oracle-free", "gvenzl/oracle-xe", "mariadb", "bitnami/mariadb", "mssql/server", "mysql", "bitnami/mysql", "postgres", or "bitnami/postgresql" | javadoc:org.springframework.boot.autoconfigure.ldap.LdapConnectionDetails[] -| Containers named "osixia/openldap" +| Containers named "osixia/openldap", "lldap/lldap" | javadoc:org.springframework.boot.autoconfigure.mongo.MongoConnectionDetails[] | Containers named "mongo" or "bitnami/mongodb" @@ -137,6 +137,92 @@ The following service connections are currently supported: +[[features.dev-services.docker-compose.ssl]] +=== SSL support + +Some images come with SSL enabled out of the box, or maybe you want to enable SSL for the container to mirror your production setup. +Spring Boot supports SSL configuration for supported service connections. +Please note that you still have to enable SSL on the service which is running inside the container yourself, this feature only configures SSL on the client side in your application. + +SSL is supported for the following service connections: + +* Cassandra +* Couchbase +* Elasticsearch +* Kafka +* MongoDB +* RabbitMQ +* Redis + +To enable SSL support for a service, you can use https://docs.docker.com/reference/compose-file/services/#labels[service labels]. + +For JKS based keystores and truststores, you can use the following container labels: + +* `org.springframework.boot.sslbundle.jks.key.alias` +* `org.springframework.boot.sslbundle.jks.key.password` +* `org.springframework.boot.sslbundle.jks.options.ciphers` +* `org.springframework.boot.sslbundle.jks.options.enabled-protocols` +* `org.springframework.boot.sslbundle.jks.protocol` + +* `org.springframework.boot.sslbundle.jks.keystore.type` +* `org.springframework.boot.sslbundle.jks.keystore.provider` +* `org.springframework.boot.sslbundle.jks.keystore.location` +* `org.springframework.boot.sslbundle.jks.keystore.password` + +* `org.springframework.boot.sslbundle.jks.truststore.type` +* `org.springframework.boot.sslbundle.jks.truststore.provider` +* `org.springframework.boot.sslbundle.jks.truststore.location` +* `org.springframework.boot.sslbundle.jks.truststore.password` + +These labels mirror the properties available for xref:reference:features/ssl.adoc#features.ssl.jks[SSL bundles]. + +For PEM based keystores and truststores, you can use the following container labels: + +* `org.springframework.boot.sslbundle.pem.key.alias` +* `org.springframework.boot.sslbundle.pem.key.password` +* `org.springframework.boot.sslbundle.pem.options.ciphers` +* `org.springframework.boot.sslbundle.pem.options.enabled-protocols` +* `org.springframework.boot.sslbundle.pem.protocol` + +* `org.springframework.boot.sslbundle.pem.keystore.type` +* `org.springframework.boot.sslbundle.pem.keystore.certificate` +* `org.springframework.boot.sslbundle.pem.keystore.private-key` +* `org.springframework.boot.sslbundle.pem.keystore.private-key-password` + +* `org.springframework.boot.sslbundle.pem.truststore.type` +* `org.springframework.boot.sslbundle.pem.truststore.certificate` +* `org.springframework.boot.sslbundle.pem.truststore.private-key` +* `org.springframework.boot.sslbundle.pem.truststore.private-key-password` + +These labels mirror the properties available for xref:reference:features/ssl.adoc#features.ssl.pem[SSL bundles]. + +The following example enables SSL for a redis container: + +[source,yaml,] +---- +services: + redis: + image: 'redis:latest' + ports: + - '6379' + secrets: + - ssl-ca + - ssl-key + - ssl-cert + command: 'redis-server --tls-port 6379 --port 0 --tls-cert-file /run/secrets/ssl-cert --tls-key-file /run/secrets/ssl-key --tls-ca-cert-file /run/secrets/ssl-ca' + labels: + - 'org.springframework.boot.sslbundle.pem.keystore.certificate=client.crt' + - 'org.springframework.boot.sslbundle.pem.keystore.private-key=client.key' + - 'org.springframework.boot.sslbundle.pem.truststore.certificate=ca.crt' +secrets: + ssl-ca: + file: 'ca.crt' + ssl-key: + file: 'server.key' + ssl-cert: + file: 'server.crt' +---- + [[features.dev-services.docker-compose.custom-images]] === Custom Images diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/developing-auto-configuration.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/developing-auto-configuration.adoc index 39d2f2568543..c278e5396a9e 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/developing-auto-configuration.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/developing-auto-configuration.adoc @@ -140,7 +140,8 @@ Providing as much type information as possible in javadoc:org.springframework.co The javadoc:org.springframework.boot.autoconfigure.condition.ConditionalOnProperty[format=annotation] annotation lets configuration be included based on a Spring Environment property. Use the `prefix` and `name` attributes to specify the property that should be checked. By default, any property that exists and is not equal to `false` is matched. -You can also create more advanced checks by using the `havingValue` and `matchIfMissing` attributes. +There is also a dedicated javadoc:org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty[format=annotation] annotation specifically made for boolean properties. +With both annotations you can also create more advanced checks by using the `havingValue` and `matchIfMissing` attributes. If multiple names are given in the `name` attribute, all of the properties have to pass the test for the condition to match. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/external-config.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/external-config.adoc index 00c96ea941c1..f0f134cff954 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/external-config.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/external-config.adoc @@ -354,11 +354,39 @@ spring: +[[features.external-config.files.env-variables]] +=== Using Environment Variables + +When running applications on a cloud platform (such as Kubernetes) you often need to read config values that the platform supplies. +You can either use environment variables for such purpose, or you can use xref:reference:features/external-config.adoc#features.external-config.files.configtree[configuration trees]. + +You can even store whole configurations in properties or yaml format in (multiline) environment variables and load them using the `env:` prefix. +Assume there's an environment variable called `MY_CONFIGURATION` with this content: + +[source,properties] +---- +my.name=Service1 +my.cluster=Cluster1 +---- + +Using the `env:` prefix it is possible to import all properties from this variable: + +[configprops,yaml] +---- +spring: + config: + import: "env:MY_CONFIGURATION" +---- + +TIP: This feature also supports xref:reference:features/external-config.adoc#features.external-config.files.importing-extensionless[specifying the extension]. +The default extension is `.properties`. + + + [[features.external-config.files.configtree]] === Using Configuration Trees -When running applications on a cloud platform (such as Kubernetes) you often need to read config values that the platform supplies. -It is not uncommon to use environment variables for such purposes, but this can have drawbacks, especially if the value is supposed to be kept secret. +Storing config values in environment variables has drawbacks, especially if the value is supposed to be kept secret. As an alternative to environment variables, many cloud platforms now allow you to map configuration into mounted data volumes. For example, Kubernetes can volume mount both https://kubernetes.io/docs/tasks/configure-pod-container/configure-pod-configmap/#populate-a-volume-with-data-stored-in-a-configmap[`ConfigMaps`] and https://kubernetes.io/docs/concepts/configuration/secret/#using-secrets-as-files-from-a-pod[`Secrets`]. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/logging.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/logging.adoc index 4af1cb203004..787e75928ea6 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/logging.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/logging.adoc @@ -641,11 +641,63 @@ logging: corpname: mycorp ---- -TIP: For more advanced customizations, you can write your own class that implements the javadoc:org.springframework.boot.logging.structured.StructuredLoggingJsonMembersCustomizer[] interface and declare it using the configprop:logging.structured.json.customizer[] property. +TIP: For more advanced customizations, you can use the javadoc:org.springframework.boot.logging.structured.StructuredLoggingJsonMembersCustomizer[] interface. +You can reference one or more implementations using the configprop:logging.structured.json.customizer[] property. You can also declare implementations by listing them in a `META-INF/spring.factories` file. +[[features.logging.structured.customizing-stack-traces]] +=== Customizing Structured Logging Stack Traces + +Complete stack traces are included in the JSON output whenever a message is logged with an exception. +This amount of information may be costly to process by your log ingestion system, so you may want to tune the way that stack traces are printed. + +To do this, you can use one or more of the following properties: + +|=== +| Property | Description + +| configprop:logging.structured.json.stacktrace.root[] +| Use `last` to print the root item last (same as Java) or `first` to print the root item first. + +| configprop:logging.structured.json.stacktrace.max-length[] +| The maximum length that should be printed + +| configprop:logging.structured.json.stacktrace.max-throwable-depth[] +| The maximum number of frames to print per stack trace (including common and suppressed frames) + +| configprop:logging.structured.json.stacktrace.include-common-frames[] +| If common frames should be included or removed + +| configprop:logging.structured.json.stacktrace.include-hashes[] +| If a hash of the stack trace should be included +|=== + +For example, the following will use root first stack traces, limit their length, and include hashes. + +[configprops,yaml] +---- +logging: + structured: + json: + stacktrace: + root: first + max-length: 1024 + include-common-frames: true + include-hashes: true +---- + +[TIP] +==== +If you need complete control over stack trace printing you can set configprop:logging.structured.json.stacktrace.printer[] to the name of a javadoc:org.springframework.boot.logging.StackTracePrinter[] implementation. +You can also set it to `logging-system` to force regular logging system stack trace output to be used. + +Your `StackTracePrinter` implementation can also include a constructor argument that accepts a javadoc:org.springframework.boot.logging.StandardStackTracePrinter[] if it wishes to apply further customization to the stack trace printer created from the properties. +==== + + + [[features.logging.structured.other-formats]] === Supporting Other Structured Logging Formats diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/task-execution-and-scheduling.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/task-execution-and-scheduling.adoc index eaceb6d0d45b..c54af72bbb98 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/task-execution-and-scheduling.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/features/task-execution-and-scheduling.adoc @@ -4,22 +4,42 @@ In the absence of an javadoc:java.util.concurrent.Executor[] bean in the context, Spring Boot auto-configures an javadoc:org.springframework.core.task.AsyncTaskExecutor[]. When virtual threads are enabled (using Java 21+ and configprop:spring.threads.virtual.enabled[] set to `true`) this will be a javadoc:org.springframework.core.task.SimpleAsyncTaskExecutor[] that uses virtual threads. Otherwise, it will be a javadoc:org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor[] with sensible defaults. -In either case, the auto-configured executor will be automatically used for: -- asynchronous task execution (`@EnableAsync`) -- Spring for GraphQL's asynchronous handling of javadoc:java.util.concurrent.Callable[] return values from controller methods -- Spring MVC's asynchronous request processing -- Spring WebFlux's blocking execution support +If a custom `Executor` bean is present, you can request Spring Boot to auto-configure an `AsyncTaskExecutor` anyway, as follows: + +[configprops,yaml] +---- +spring: + task: + execution: + mode: force +---- + +The auto-configured executor will be automatically used for: + +- Asynchronous task execution (`@EnableAsync`), unless an javadoc:org.springframework.scheduling.annotation.AsyncConfigurer[] bean is present. +- Spring for GraphQL's asynchronous handling of javadoc:java.util.concurrent.Callable[] return values from controller methods. +- Spring MVC's asynchronous request processing. +- Spring WebFlux's blocking execution support. [TIP] ==== If you have defined a custom javadoc:java.util.concurrent.Executor[] in the context, both regular task execution (that is javadoc:org.springframework.scheduling.annotation.EnableAsync[format=annotation]) and Spring for GraphQL will use it. -However, the Spring MVC and Spring WebFlux support will only use it if it is an javadoc:org.springframework.core.task.AsyncTaskExecutor[] implementation (named `applicationTaskExecutor`). -Depending on your target arrangement, you could change your javadoc:java.util.concurrent.Executor[] into an javadoc:org.springframework.core.task.AsyncTaskExecutor[] or define both an javadoc:org.springframework.core.task.AsyncTaskExecutor[] and an javadoc:org.springframework.scheduling.annotation.AsyncConfigurer[] wrapping your custom javadoc:java.util.concurrent.Executor[]. +However, the Spring MVC and Spring WebFlux support will only use it if it is an javadoc:org.springframework.core.task.AsyncTaskExecutor[] implementation named `applicationTaskExecutor`. -The auto-configured javadoc:org.springframework.boot.task.ThreadPoolTaskExecutorBuilder[] allows you to easily create instances that reproduce what the auto-configuration does by default. +Depending on your target arrangement, you could set configprop:spring.task.execution.mode[] to `force` to auto-configure an `applicationTaskExecutor`, change your javadoc:java.util.concurrent.Executor[] into an javadoc:org.springframework.core.task.AsyncTaskExecutor[] or define both an javadoc:org.springframework.core.task.AsyncTaskExecutor[] and an javadoc:org.springframework.scheduling.annotation.AsyncConfigurer[] wrapping your custom javadoc:java.util.concurrent.Executor[]. + +Another option is to define those beans explicitly. +The auto-configured javadoc:org.springframework.boot.task.ThreadPoolTaskExecutorBuilder[] or javadoc:org.springframework.boot.task.SimpleAsyncTaskExecutorBuilder[] allow you to easily create instances that reproduce what the auto-configuration does by default. ==== +[NOTE] +==== +If multiple javadoc:java.util.concurrent.Executor[] beans are defined with configprop:spring.task.execution.mode[] to `force`, all the supported integrations look for a bean named `applicationTaskExecutor`. +If the auto-configured `AsyncTaskExecutor` is not defined, only regular task execution fallbacks to a bean named `taskExecutor` to match Spring Framework's behavior. +==== + + When a javadoc:org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor[] is auto-configured, the thread pool uses 8 core threads that can grow and shrink according to the load. Those default settings can be fine-tuned using the `spring.task.execution` namespace, as shown in the following example: diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/io/rest-client.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/io/rest-client.adoc index b0a8045b3476..f9264f33433b 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/io/rest-client.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/io/rest-client.adoc @@ -52,6 +52,45 @@ You can learn more about the {url-spring-framework-docs}/web/webflux-webclient/c +[[io.rest-client.webclient.configuration]] +=== Global HTTP Connector Configuration + +If the auto-detected javadoc:org.springframework.http.client.reactive.ClientHttpConnector[] does not meet your needs, you can use the configprop:spring.http.reactiveclient.settings.connector[] property to pick a specific connector. +For example, if you have Reactor Netty on your classpath, but you prefer Jetty's javadoc:org.eclipse.jetty.client.HttpClient[] you can add the following: + +[configprops,yaml] +---- +spring: + http: + reactiveclient: + settings: + connector: jetty +---- + +You can also set properties to change defaults that will be applied to all reactive connectors. +For example, you may want to change timeouts and if redirects are followed: + +[configprops,yaml] +---- +spring: + http: + reactiveclient: + settings: + connect-timeout: 2s + read-timeout: 1s + redirects: dont-follow +---- + +For more complex customizations, you can use javadoc:org.springframework.boot.autoconfigure.http.client.reactive.ClientHttpConnectorBuilderCustomizer[] or declare your own javadoc:org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder[] bean which will cause auto-configuration to back off. +This can be useful when you need to customize some of the internals of the underlying HTTP library. + +For example, the following will use a JDK client configured with a specific javadoc:java.net.ProxySelector[]: + +include-code::MyConnectorHttpConfiguration[] + + + + [[io.rest-client.webclient.customization]] === WebClient Customization @@ -233,7 +272,7 @@ spring: redirects: dont-follow ---- -For more complex customizations, you can declare your own javadoc:org.springframework.boot.http.client.ClientHttpRequestFactoryBuilder[] bean which will cause auto-configuration to back off. +For more complex customizations, you can use javadoc:org.springframework.boot.autoconfigure.http.client.ClientHttpRequestFactoryBuilderCustomizer[] or declare your own javadoc:org.springframework.boot.http.client.ClientHttpRequestFactoryBuilder[] bean which will cause auto-configuration to back off. This can be useful when you need to customize some of the internals of the underlying HTTP library. For example, the following will use a JDK client configured with a specific javadoc:java.net.ProxySelector[]: diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/testing/spring-boot-applications.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/testing/spring-boot-applications.adoc index 508b2b5c2cdb..f9b8ac9a02db 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/testing/spring-boot-applications.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/testing/spring-boot-applications.adoc @@ -319,7 +319,7 @@ TIP: A list of the auto-configuration settings that are enabled by javadoc:org.s TIP: If you need to register extra components, such as the Jackson javadoc:com.fasterxml.jackson.databind.Module[], you can import additional configuration classes by using javadoc:org.springframework.context.annotation.Import[format=annotation] on your test. -Often, javadoc:org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest[format=annotation] is limited to a single controller and is used in combination with javadoc:org.springframework.boot.test.mock.mockito.MockBean[format=annotation] to provide mock implementations for required collaborators. +Often, javadoc:org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest[format=annotation] is limited to a single controller and is used in combination with javadoc:org.springframework.test.context.bean.override.mockito.MockitoBean[format=annotation] to provide mock implementations for required collaborators. javadoc:org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest[format=annotation] also auto-configures javadoc:org.springframework.test.web.servlet.MockMvc[]. Mock MVC offers a powerful way to quickly test MVC controllers without needing to start a full HTTP server. @@ -363,7 +363,7 @@ TIP: A list of the auto-configurations that are enabled by javadoc:org.springfra TIP: If you need to register extra components, such as Jackson javadoc:com.fasterxml.jackson.databind.Module[], you can import additional configuration classes using javadoc:org.springframework.context.annotation.Import[format=annotation] on your test. -Often, javadoc:org.springframework.boot.test.autoconfigure.web.reactive.WebFluxTest[format=annotation] is limited to a single controller and used in combination with the javadoc:org.springframework.boot.test.mock.mockito.MockBean[format=annotation] annotation to provide mock implementations for required collaborators. +Often, javadoc:org.springframework.boot.test.autoconfigure.web.reactive.WebFluxTest[format=annotation] is limited to a single controller and used in combination with the javadoc:org.springframework.test.context.bean.override.mockito.MockitoBean[format=annotation] annotation to provide mock implementations for required collaborators. javadoc:org.springframework.boot.test.autoconfigure.web.reactive.WebFluxTest[format=annotation] also auto-configures {url-spring-framework-docs}/testing/webtestclient.html[`WebTestClient`], which offers a powerful way to quickly test WebFlux controllers without needing to start a full HTTP server. @@ -432,7 +432,7 @@ javadoc:org.springframework.boot.context.properties.EnableConfigurationPropertie TIP: A list of the auto-configurations that are enabled by javadoc:org.springframework.boot.test.autoconfigure.graphql.GraphQlTest[format=annotation] can be xref:appendix:test-auto-configuration/index.adoc[found in the appendix]. -Often, javadoc:org.springframework.boot.test.autoconfigure.graphql.GraphQlTest[format=annotation] is limited to a set of controllers and used in combination with the javadoc:org.springframework.boot.test.mock.mockito.MockBean[format=annotation] annotation to provide mock implementations for required collaborators. +Often, javadoc:org.springframework.boot.test.autoconfigure.graphql.GraphQlTest[format=annotation] is limited to a set of controllers and used in combination with the javadoc:org.springframework.test.context.bean.override.mockito.MockitoBean[format=annotation] annotation to provide mock implementations for required collaborators. include-code::GreetingControllerTests[] diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/testing/testcontainers.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/testing/testcontainers.adoc index 00bdcf606579..350794cf2f06 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/testing/testcontainers.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/testing/testcontainers.adoc @@ -62,6 +62,9 @@ The following service connection factories are provided in the `spring-boot-test | javadoc:org.springframework.boot.autoconfigure.kafka.KafkaConnectionDetails[] | Containers of type javadoc:org.testcontainers.kafka.KafkaContainer[], javadoc:org.testcontainers.kafka.ConfluentKafkaContainer[] or javadoc:org.testcontainers.redpanda.RedpandaContainer[] +| javadoc:org.springframework.boot.autoconfigure.ldap.LdapConnectionDetails[] +| Containers named "osixia/openldap" or of type javadoc:org.testcontainers.ldap.LLdapContainer[] + | javadoc:org.springframework.boot.autoconfigure.liquibase.LiquibaseConnectionDetails[] | Containers of type javadoc:{url-testcontainers-jdbc-javadoc}/org.testcontainers.containers.JdbcDatabaseContainer[] @@ -122,6 +125,36 @@ If you are using the Docker image `registry.mycompany.com/mirror/myredis`, you'd +[[testing.testcontainers.service-connections.ssl]] +=== SSL with Service Connections + +You can use the javadoc:org.springframework.boot.testcontainers.service.connection.Ssl[format=annotation], javadoc:org.springframework.boot.testcontainers.service.connection.JksKeyStore[format=annotation], javadoc:org.springframework.boot.testcontainers.service.connection.JksTrustStore[format=annotation], javadoc:org.springframework.boot.testcontainers.service.connection.PemKeyStore[format=annotation] and javadoc:org.springframework.boot.testcontainers.service.connection.PemTrustStore[format=annotation] annotations on a supported container to enable SSL support for that service connection. +Please note that you still have to enable SSL on the service which is running inside the Testcontainer yourself, the annotations only configure SSL on the client side in your application. + +include-code::MyRedisWithSslIntegrationTests[] + +The above code uses the javadoc:org.springframework.boot.testcontainers.service.connection.PemKeyStore[format=annotation] annotation to load the client certificate and key into the keystore and the and javadoc:org.springframework.boot.testcontainers.service.connection.PemTrustStore[format=annotation] annotation to load the CA certificate into the truststore. +This will authenticate the client against the server, and the CA certificate in the truststore makes sure that the server certificate is valid and trusted. + +The `SecureRedisContainer` in this example is a custom subclass of `RedisContainer` which copies certificates to the correct places and invokes `redis-server` with commandline parameters enabling SSL. + +The SSL annotations are supported for the following service connections: + +* Cassandra +* Couchbase +* Elasticsearch +* Kafka +* MongoDB +* RabbitMQ +* Redis + +The `ElasticsearchContainer` additionally supports automatic detection of server side SSL. +To use this feature, annotate the container with javadoc:org.springframework.boot.testcontainers.service.connection.Ssl[format=annotation], as seen in the following example, and Spring Boot takes care of the client side SSL configuration for you: + +include-code::MyElasticsearchWithSslIntegrationTests[] + + + [[testing.testcontainers.dynamic-properties]] == Dynamic Properties diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/using/devtools.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/using/devtools.adoc index 1aff16777948..d1499cc6a235 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/using/devtools.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/using/devtools.adoc @@ -78,7 +78,7 @@ NOTE: If you do not want property defaults to be applied you can set configprop: Because you need more information about web requests while developing Spring MVC and Spring WebFlux applications, developer tools suggests you to enable `DEBUG` logging for the `web` logging group. This will give you information about the incoming request, which handler is processing it, the response outcome, and other details. -If you wish to log all request details (including potentially sensitive information), you can turn on the configprop:spring.mvc.log-request-details[] or configprop:spring.codec.log-request-details[] configuration properties. +If you wish to log all request details (including potentially sensitive information), you can turn on the configprop:spring.mvc.log-request-details[] or configprop:spring.http.codecs.log-request-details[] configuration properties. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/reactive.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/reactive.adoc index 85277085b8be..ef92b27b99c7 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/reactive.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/reactive.adoc @@ -87,7 +87,7 @@ When not configured, the following defaults are used: Spring WebFlux uses the javadoc:org.springframework.http.codec.HttpMessageReader[] and javadoc:org.springframework.http.codec.HttpMessageWriter[] interfaces to convert HTTP requests and responses. They are configured with javadoc:org.springframework.http.codec.CodecConfigurer[] to have sensible defaults by looking at the libraries available in your classpath. -Spring Boot provides dedicated configuration properties for codecs, `+spring.codec.*+`. +Spring Boot provides dedicated configuration properties for codecs, `+spring.http.codecs.*+`. It also applies further customization by using javadoc:org.springframework.boot.web.codec.CodecCustomizer[] instances. For example, `+spring.jackson.*+` configuration keys are applied to the Jackson codec. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/servlet.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/servlet.adoc index b431a1519fb1..ba3088f09176 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/servlet.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/servlet.adoc @@ -529,11 +529,14 @@ In the case of multiple servlet beans, the bean name is used as a path prefix. Filters map to `+/*+`. If convention-based mapping is not flexible enough, you can use the javadoc:org.springframework.boot.web.servlet.ServletRegistrationBean[], javadoc:org.springframework.boot.web.servlet.FilterRegistrationBean[], and javadoc:org.springframework.boot.web.servlet.ServletListenerRegistrationBean[] classes for complete control. +If you prefer annotations over javadoc:org.springframework.boot.web.servlet.ServletRegistrationBean[] and javadoc:org.springframework.boot.web.servlet.FilterRegistrationBean[], you can also use javadoc:org.springframework.boot.web.servlet.ServletRegistration[format=annotation] and +javadoc:org.springframework.boot.web.servlet.FilterRegistration[format=annotation] as an alternative. It is usually safe to leave filter beans unordered. If a specific order is required, you should annotate the javadoc:jakarta.servlet.Filter[] with javadoc:org.springframework.core.annotation.Order[format=annotation] or make it implement javadoc:org.springframework.core.Ordered[]. You cannot configure the order of a javadoc:jakarta.servlet.Filter[] by annotating its bean method with javadoc:org.springframework.core.annotation.Order[format=annotation]. If you cannot change the javadoc:jakarta.servlet.Filter[] class to add javadoc:org.springframework.core.annotation.Order[format=annotation] or implement javadoc:org.springframework.core.Ordered[], you must define a javadoc:org.springframework.boot.web.servlet.FilterRegistrationBean[] for the javadoc:jakarta.servlet.Filter[] and set the registration bean's order using the `setOrder(int)` method. +Or, if you prefer annotations, you can also use javadoc:org.springframework.boot.web.servlet.FilterRegistration[format=annotation] and set the `order` attribute. Avoid configuring a filter that reads the request body at `Ordered.HIGHEST_PRECEDENCE`, since it might go against the character encoding configuration of your application. If a servlet filter wraps the request, it should be configured with an order that is less than or equal to `OrderedFilter.REQUEST_WRAPPER_FILTER_MAX_ORDER`. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/spring-graphql.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/spring-graphql.adoc index fbddcbc82246..0add65c8a45c 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/spring-graphql.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/spring-graphql.adoc @@ -94,7 +94,7 @@ are detected by Spring Boot and considered as candidates for javadoc:graphql.sch The GraphQL HTTP endpoint is at HTTP POST `/graphql` by default. It also supports the `"text/event-stream"` media type over Server Sent Events for subscriptions only. -The path can be customized with configprop:spring.graphql.path[]. +The path can be customized with configprop:spring.graphql.http.path[]. TIP: The HTTP endpoint for both Spring MVC and Spring WebFlux is provided by a `RouterFunction` bean with an javadoc:org.springframework.core.annotation.Order[format=annotation] of `0`. If you define your own `RouterFunction` beans, you may want to add appropriate javadoc:org.springframework.core.annotation.Order[format=annotation] annotations to ensure that they are sorted correctly. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/spring-security.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/spring-security.adoc index 3fe36cc92e25..117752ee09dd 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/spring-security.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/reference/pages/web/spring-security.adoc @@ -35,7 +35,7 @@ You can provide a different javadoc:org.springframework.security.authentication. == MVC Security The default security configuration is implemented in javadoc:org.springframework.boot.autoconfigure.security.servlet.SecurityAutoConfiguration[] and javadoc:org.springframework.boot.autoconfigure.security.servlet.UserDetailsServiceAutoConfiguration[]. -javadoc:org.springframework.boot.autoconfigure.security.servlet.SecurityAutoConfiguration[] imports `SpringBootWebSecurityConfiguration` for web security and javadoc:org.springframework.boot.autoconfigure.security.servlet.UserDetailsServiceAutoConfiguration[] configures authentication, which is also relevant in non-web applications. +javadoc:org.springframework.boot.autoconfigure.security.servlet.SecurityAutoConfiguration[] imports `SpringBootWebSecurityConfiguration` for web security and javadoc:org.springframework.boot.autoconfigure.security.servlet.UserDetailsServiceAutoConfiguration[] for authentication. To completely switch off the default web application security configuration, including Actuator security, or to combine multiple Spring Security components such as OAuth2 Client and Resource Server, add a bean of type javadoc:org.springframework.security.web.SecurityFilterChain[] (doing so does not disable the javadoc:org.springframework.security.core.userdetails.UserDetailsService[] configuration). To also switch off the javadoc:org.springframework.security.core.userdetails.UserDetailsService[] configuration, add a bean of type javadoc:org.springframework.security.core.userdetails.UserDetailsService[], javadoc:org.springframework.security.authentication.AuthenticationProvider[], or javadoc:org.springframework.security.authentication.AuthenticationManager[]. @@ -59,11 +59,12 @@ javadoc:org.springframework.boot.autoconfigure.security.servlet.PathRequest[] ca == WebFlux Security Similar to Spring MVC applications, you can secure your WebFlux applications by adding the `spring-boot-starter-security` dependency. -The default security configuration is implemented in javadoc:org.springframework.boot.autoconfigure.security.reactive.ReactiveSecurityAutoConfiguration[] and javadoc:org.springframework.boot.autoconfigure.security.servlet.UserDetailsServiceAutoConfiguration[]. -javadoc:org.springframework.boot.autoconfigure.security.reactive.ReactiveSecurityAutoConfiguration[] imports `WebFluxSecurityConfiguration` for web security and javadoc:org.springframework.boot.autoconfigure.security.servlet.UserDetailsServiceAutoConfiguration[] configures authentication, which is also relevant in non-web applications. +The default security configuration is implemented in javadoc:org.springframework.boot.autoconfigure.security.reactive.ReactiveSecurityAutoConfiguration[] and javadoc:org.springframework.boot.autoconfigure.security.reactive.ReactiveUserDetailsServiceAutoConfiguration[]. +javadoc:org.springframework.boot.autoconfigure.security.reactive.ReactiveSecurityAutoConfiguration[] imports `WebFluxSecurityConfiguration` for web security and javadoc:org.springframework.boot.autoconfigure.security.reactive.UserDetailsServiceAutoConfiguration[] for authentication. +In addition to reactive web applications, the latter is also auto-configured when RSocket is in use. -To completely switch off the default web application security configuration, including Actuator security, add a bean of type javadoc:org.springframework.security.web.server.WebFilterChainProxy[] (doing so does not disable the javadoc:org.springframework.security.core.userdetails.UserDetailsService[] configuration). -To also switch off the javadoc:org.springframework.security.core.userdetails.UserDetailsService[] configuration, add a bean of type javadoc:org.springframework.security.core.userdetails.ReactiveUserDetailsService[] or javadoc:org.springframework.security.authentication.ReactiveAuthenticationManager[]. +To completely switch off the default web application security configuration, including Actuator security, add a bean of type javadoc:org.springframework.security.web.server.WebFilterChainProxy[] (doing so does not disable the javadoc:org.springframework.security.core.userdetails.ReactiveUserDetailsService[] configuration). +To also switch off the javadoc:org.springframework.security.core.userdetails.ReactiveUserDetailsService[] configuration, add a bean of type javadoc:org.springframework.security.core.userdetails.ReactiveUserDetailsService[] or javadoc:org.springframework.security.authentication.ReactiveAuthenticationManager[]. The auto-configuration will also back off when any of the following Spring Security modules is on the classpath: diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/specification/pages/configuration-metadata/annotation-processor.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/specification/pages/configuration-metadata/annotation-processor.adoc index 04e9d53d3e7c..941d73f9abb5 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/specification/pages/configuration-metadata/annotation-processor.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/specification/pages/configuration-metadata/annotation-processor.adoc @@ -160,12 +160,14 @@ TIP: This has no effect on collections and maps, as those types are automaticall == Adding Additional Metadata Spring Boot's configuration file handling is quite flexible, and it is often the case that properties may exist that are not bound to a javadoc:org.springframework.boot.context.properties.ConfigurationProperties[format=annotation] bean. -You may also need to tune some attributes of an existing key. +You may also need to tune some attributes of an existing key or to ignore the key altogether. To support such cases and let you provide custom "hints", the annotation processor automatically merges items from `META-INF/additional-spring-configuration-metadata.json` into the main metadata file. If you refer to a property that has been detected automatically, the description, default value, and deprecation information are overridden, if specified. If the manual property declaration is not identified in the current module, it is added as a new property. The format of the `additional-spring-configuration-metadata.json` file is exactly the same as the regular `spring-configuration-metadata.json`. +The items contained in the "`ignored.properties`" section are removed from the "`properties`" section of the generated `spring-configuration-metadata.json` file. + The additional properties file is optional. If you do not have any additional properties, do not add the file. diff --git a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/specification/pages/configuration-metadata/format.adoc b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/specification/pages/configuration-metadata/format.adoc index db8964b14767..ae560f89901b 100644 --- a/spring-boot-project/spring-boot-docs/src/docs/antora/modules/specification/pages/configuration-metadata/format.adoc +++ b/spring-boot-project/spring-boot-docs/src/docs/antora/modules/specification/pages/configuration-metadata/format.adoc @@ -2,7 +2,7 @@ = Metadata Format Configuration metadata files are located inside jars under `META-INF/spring-configuration-metadata.json`. -They use a JSON format with items categorized under either "`groups`" or "`properties`" and additional values hints categorized under "hints", as shown in the following example: +They use a JSON format with items categorized under either "`groups`" or "`properties`", additional values hints categorized under "`hints`", and ignored items under "`ignored`" as shown in the following example: [source,json] ---- @@ -63,7 +63,15 @@ They use a JSON format with items categorized under either "`groups`" or "`prope } ] } -]} + ... +],"ignored": { + "properties": [ + { + "name": "server.ignored" + } + ... + ] +}} ---- Each "`property`" is a configuration item that the user specifies with a given value. @@ -82,9 +90,12 @@ For example, the `server.port` and `server.address` properties are part of the ` NOTE: It is not required that every "`property`" has a "`group`". Some properties might exist in their own right. -Finally, "`hints`" are additional information used to assist the user in configuring a given property. +The "`hints`" are additional information used to assist the user in configuring a given property. For example, when a developer is configuring the configprop:spring.jpa.hibernate.ddl-auto[] property, a tool can use the hints to offer some auto-completion help for the `none`, `validate`, `update`, `create`, and `create-drop` values. +Finally, "`ignored`" is for items which have been deliberately ignored. +The content of this section usually comes from the xref:specification:configuration-metadata/annotation-processor.adoc#appendix.configuration-metadata.annotation-processor.adding-additional-metadata[additional metadata]. + [[appendix.configuration-metadata.format.group]] @@ -292,6 +303,36 @@ The JSON object contained in the `providers` attribute of each `hint` element ca +[[appendix.configuration-metadata.format.ignored]] +== Ignored Attributes + +The `ignored` object can contain the attributes shown in the following table: + +[cols="1,1,4"] +|=== +| Name | Type | Purpose + +| `properties` +| ItemIgnore[] +| A list of ignored properties as defined by the ItemIgnore object (described in the next table). Each entry defines the name of the ignored property. + +|=== + +The JSON object contained in the `properties` attribute of each `ignored` element can contain the attributes described in the following table: + +[cols="1,1,4"] +|=== +| Name | Type | Purpose + +| `name` +| String +| The full name of the property to ignore. +Names are in lower-case period-separated form (such as `spring.mvc.servlet.path`). +This attribute is mandatory. + +|=== + + [[appendix.configuration-metadata.format.repeated-items]] == Repeated Metadata Items diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyMessagingProperties.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyMessagingProperties.java index e03d8b131d1d..60f147b20320 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyMessagingProperties.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyMessagingProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,7 +22,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; -@ConfigurationProperties(prefix = "my.messaging") +@ConfigurationProperties("my.messaging") public class MyMessagingProperties { private List addresses = new ArrayList<>(Arrays.asList("a", "b")); diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyServerProperties.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyServerProperties.java index 381ce63216fc..a12403c9e060 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyServerProperties.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyServerProperties.java @@ -18,7 +18,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; -@ConfigurationProperties(prefix = "my.server") +@ConfigurationProperties("my.server") public class MyServerProperties { /** diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/nestedproperties/MyServerProperties.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/nestedproperties/MyServerProperties.java index fbb00298d855..833630c5bda4 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/nestedproperties/MyServerProperties.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/appendix/configurationmetadata/annotationprocessor/automaticmetadatageneration/nestedproperties/MyServerProperties.java @@ -18,7 +18,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; -@ConfigurationProperties(prefix = "my.server") +@ConfigurationProperties("my.server") public class MyServerProperties { private String name; diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/MyPersonProperties.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/MyPersonProperties.java index 74fb9d6bd55f..0ad4475a1f36 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/MyPersonProperties.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/MyPersonProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,7 +18,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; -@ConfigurationProperties(prefix = "my.main-project.person") +@ConfigurationProperties("my.main-project.person") public class MyPersonProperties { private String firstName; diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/mapsfromenvironmentvariables/MyMapsProperties.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/mapsfromenvironmentvariables/MyMapsProperties.java index 7abc11edb290..f769c53e1ff0 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/mapsfromenvironmentvariables/MyMapsProperties.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/mapsfromenvironmentvariables/MyMapsProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,7 +21,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; -@ConfigurationProperties(prefix = "my.props") +@ConfigurationProperties("my.props") public class MyMapsProperties { private final Map values = new HashMap<>(); diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/thirdpartyconfiguration/ThirdPartyConfiguration.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/thirdpartyconfiguration/ThirdPartyConfiguration.java index b63271555683..33b5dad07057 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/thirdpartyconfiguration/ThirdPartyConfiguration.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/thirdpartyconfiguration/ThirdPartyConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,7 @@ public class ThirdPartyConfiguration { @Bean - @ConfigurationProperties(prefix = "another") + @ConfigurationProperties("another") public AnotherComponent anotherComponent() { return new AnotherComponent(); } diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/application/customizetheenvironmentorapplicationcontext/MyEnvironmentPostProcessor.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/application/customizetheenvironmentorapplicationcontext/MyEnvironmentPostProcessor.java index f2ef2cfbb54d..ea5fde11f220 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/application/customizetheenvironmentorapplicationcontext/MyEnvironmentPostProcessor.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/application/customizetheenvironmentorapplicationcontext/MyEnvironmentPostProcessor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,7 @@ public void postProcessEnvironment(ConfigurableEnvironment environment, SpringAp } private PropertySource loadYaml(Resource path) { - Assert.isTrue(path.exists(), () -> "Resource " + path + " does not exist"); + Assert.isTrue(path.exists(), () -> "'path' [%s] must exist".formatted(path)); try { return this.loader.load("custom-resource", path).get(0); } diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/dataaccess/configurecustomdatasource/custom/MyDataSourceConfiguration.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/dataaccess/configurecustomdatasource/custom/MyDataSourceConfiguration.java index 72104f971ab3..bc7392448fdf 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/dataaccess/configurecustomdatasource/custom/MyDataSourceConfiguration.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/dataaccess/configurecustomdatasource/custom/MyDataSourceConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,7 @@ public class MyDataSourceConfiguration { @Bean - @ConfigurationProperties(prefix = "app.datasource") + @ConfigurationProperties("app.datasource") public SomeDataSource dataSource() { return new SomeDataSource(); } diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/security/enablehttps/MySecurityConfig.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/security/enablehttps/MySecurityConfig.java index 4b007e241d9b..6fdaf47566d7 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/security/enablehttps/MySecurityConfig.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/howto/security/enablehttps/MySecurityConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,6 +18,7 @@ import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import org.springframework.security.config.Customizer; import org.springframework.security.config.annotation.web.builders.HttpSecurity; import org.springframework.security.web.SecurityFilterChain; @@ -27,7 +28,7 @@ public class MySecurityConfig { @Bean public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception { // Customize the application security ... - http.requiresChannel((channel) -> channel.anyRequest().requiresSecure()); + http.redirectToHttps(Customizer.withDefaults()); return http.build(); } diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/io/restclient/webclient/configuration/MyConnectorHttpConfiguration.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/io/restclient/webclient/configuration/MyConnectorHttpConfiguration.java new file mode 100644 index 000000000000..d9c81dd6582e --- /dev/null +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/io/restclient/webclient/configuration/MyConnectorHttpConfiguration.java @@ -0,0 +1,33 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.docs.io.restclient.webclient.configuration; + +import java.net.ProxySelector; + +import org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +@Configuration(proxyBeanMethods = false) +public class MyConnectorHttpConfiguration { + + @Bean + ClientHttpConnectorBuilder clientHttpConnectorBuilder(ProxySelector proxySelector) { + return ClientHttpConnectorBuilder.jdk().withHttpClientCustomizer((builder) -> builder.proxy(proxySelector)); + } + +} diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyProperties.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyProperties.java index e0b25257b01f..d06b26ca62e0 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyProperties.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,7 +19,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.boot.context.properties.NestedConfigurationProperty; -@ConfigurationProperties(prefix = "my.properties") +@ConfigurationProperties("my.properties") public class MyProperties { private String name; diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesCtor.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesCtor.java index 861947e1f697..884af960d7d8 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesCtor.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesCtor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,7 +19,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.boot.context.properties.NestedConfigurationProperty; -@ConfigurationProperties(prefix = "my.properties") +@ConfigurationProperties("my.properties") public class MyPropertiesCtor { private final String name; diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesRecord.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesRecord.java index 8322d7d83af0..0aee00b22684 100644 --- a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesRecord.java +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesRecord.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,7 +19,7 @@ import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.boot.context.properties.NestedConfigurationProperty; -@ConfigurationProperties(prefix = "my.properties") +@ConfigurationProperties("my.properties") public record MyPropertiesRecord(String name, @NestedConfigurationProperty Nested nested) { } diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/testing/testcontainers/serviceconnections/ssl/MyElasticsearchWithSslIntegrationTests.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/testing/testcontainers/serviceconnections/ssl/MyElasticsearchWithSslIntegrationTests.java new file mode 100644 index 000000000000..919eaaedb6f2 --- /dev/null +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/testing/testcontainers/serviceconnections/ssl/MyElasticsearchWithSslIntegrationTests.java @@ -0,0 +1,49 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.docs.testing.testcontainers.serviceconnections.ssl; + +import org.junit.jupiter.api.Test; +import org.testcontainers.elasticsearch.ElasticsearchContainer; +import org.testcontainers.junit.jupiter.Container; +import org.testcontainers.junit.jupiter.Testcontainers; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.data.elasticsearch.DataElasticsearchTest; +import org.springframework.boot.testcontainers.service.connection.ServiceConnection; +import org.springframework.boot.testcontainers.service.connection.Ssl; +import org.springframework.data.elasticsearch.client.elc.ElasticsearchTemplate; + +@Testcontainers +@DataElasticsearchTest +class MyElasticsearchWithSslIntegrationTests { + + @Ssl + @Container + @ServiceConnection + static ElasticsearchContainer elasticsearch = new ElasticsearchContainer( + "docker.elastic.co/elasticsearch/elasticsearch:8.17.2"); + + @Autowired + @SuppressWarnings("unused") + private ElasticsearchTemplate elasticsearchTemplate; + + @Test + void testElasticsearch() { + // ... + } + +} diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/testing/testcontainers/serviceconnections/ssl/MyRedisWithSslIntegrationTests.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/testing/testcontainers/serviceconnections/ssl/MyRedisWithSslIntegrationTests.java new file mode 100644 index 000000000000..e052cee6872b --- /dev/null +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/testing/testcontainers/serviceconnections/ssl/MyRedisWithSslIntegrationTests.java @@ -0,0 +1,50 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.docs.testing.testcontainers.serviceconnections.ssl; + +import com.redis.testcontainers.RedisContainer; +import org.junit.jupiter.api.Test; +import org.testcontainers.junit.jupiter.Container; +import org.testcontainers.junit.jupiter.Testcontainers; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.boot.testcontainers.service.connection.PemKeyStore; +import org.springframework.boot.testcontainers.service.connection.PemTrustStore; +import org.springframework.boot.testcontainers.service.connection.ServiceConnection; +import org.springframework.data.redis.core.RedisOperations; + +@Testcontainers +@SpringBootTest +class MyRedisWithSslIntegrationTests { + + @Container + @ServiceConnection + @PemKeyStore(certificate = "classpath:client.crt", privateKey = "classpath:client.key") + @PemTrustStore("classpath:ca.crt") + static RedisContainer redis = new SecureRedisContainer("redis:latest"); + + @Autowired + @SuppressWarnings("unused") + private RedisOperations operations; + + @Test + void testRedis() { + // ... + } + +} diff --git a/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/testing/testcontainers/serviceconnections/ssl/SecureRedisContainer.java b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/testing/testcontainers/serviceconnections/ssl/SecureRedisContainer.java new file mode 100644 index 000000000000..b0a251c7ac46 --- /dev/null +++ b/spring-boot-project/spring-boot-docs/src/main/java/org/springframework/boot/docs/testing/testcontainers/serviceconnections/ssl/SecureRedisContainer.java @@ -0,0 +1,27 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.docs.testing.testcontainers.serviceconnections.ssl; + +import com.redis.testcontainers.RedisContainer; + +class SecureRedisContainer extends RedisContainer { + + SecureRedisContainer(String dockerImageName) { + super(dockerImageName); + } + +} diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyMessagingProperties.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyMessagingProperties.kt index a87d99e960d0..c074a8362775 100644 --- a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyMessagingProperties.kt +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyMessagingProperties.kt @@ -19,7 +19,7 @@ package org.springframework.boot.docs.configurationmetadata.annotationprocessor. import org.springframework.boot.context.properties.ConfigurationProperties import java.util.Arrays -@ConfigurationProperties(prefix = "my.messaging") +@ConfigurationProperties("my.messaging") class MyMessagingProperties( val addresses: List = ArrayList(Arrays.asList("a", "b")), diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyServerProperties.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyServerProperties.kt index 11b8cf82b1f9..27f4382bcb9d 100644 --- a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyServerProperties.kt +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/MyServerProperties.kt @@ -18,7 +18,7 @@ package org.springframework.boot.docs.configurationmetadata.annotationprocessor. import org.springframework.boot.context.properties.ConfigurationProperties -@ConfigurationProperties(prefix = "my.server") +@ConfigurationProperties("my.server") class MyServerProperties( /** diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/nestedproperties/MyServerProperties.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/nestedproperties/MyServerProperties.kt index d2119679086b..9e7c1fad0592 100644 --- a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/nestedproperties/MyServerProperties.kt +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/configurationmetadata/annotationprocessor/automaticmetadatageneration/nestedproperties/MyServerProperties.kt @@ -18,7 +18,7 @@ package org.springframework.boot.docs.configurationmetadata.annotationprocessor. import org.springframework.boot.context.properties.ConfigurationProperties -@ConfigurationProperties(prefix = "my.server") +@ConfigurationProperties("my.server") class MyServerProperties( var name: String, var host: Host) { diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/MyPersonProperties.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/MyPersonProperties.kt index 6be11c0365ff..b903654dc8ab 100644 --- a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/MyPersonProperties.kt +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/MyPersonProperties.kt @@ -18,7 +18,7 @@ package org.springframework.boot.docs.features.externalconfig.typesafeconfigurat import org.springframework.boot.context.properties.ConfigurationProperties -@ConfigurationProperties(prefix = "my.main-project.person") +@ConfigurationProperties("my.main-project.person") class MyPersonProperties { var firstName: String? = null diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/mapsfromenvironmentvariables/MyMapsProperties.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/mapsfromenvironmentvariables/MyMapsProperties.kt index f58251e968cb..d537175b8142 100644 --- a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/mapsfromenvironmentvariables/MyMapsProperties.kt +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/relaxedbinding/mapsfromenvironmentvariables/MyMapsProperties.kt @@ -2,7 +2,7 @@ package org.springframework.boot.docs.features.externalconfig.typesafeconfigurat import org.springframework.boot.context.properties.ConfigurationProperties -@ConfigurationProperties(prefix = "my.props") +@ConfigurationProperties("my.props") class MyMapsProperties { val values: Map = HashMap() diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/thirdpartyconfiguration/ThirdPartyConfiguration.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/thirdpartyconfiguration/ThirdPartyConfiguration.kt index ae4039cd62c9..6949f4a1e504 100644 --- a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/thirdpartyconfiguration/ThirdPartyConfiguration.kt +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/features/externalconfig/typesafeconfigurationproperties/thirdpartyconfiguration/ThirdPartyConfiguration.kt @@ -24,7 +24,7 @@ import org.springframework.context.annotation.Configuration class ThirdPartyConfiguration { @Bean - @ConfigurationProperties(prefix = "another") + @ConfigurationProperties("another") fun anotherComponent(): AnotherComponent = AnotherComponent() } diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/howto/dataaccess/configurecustomdatasource/custom/MyDataSourceConfiguration.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/howto/dataaccess/configurecustomdatasource/custom/MyDataSourceConfiguration.kt index da7e2d11fa37..2e5159736ce0 100644 --- a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/howto/dataaccess/configurecustomdatasource/custom/MyDataSourceConfiguration.kt +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/howto/dataaccess/configurecustomdatasource/custom/MyDataSourceConfiguration.kt @@ -24,7 +24,7 @@ import org.springframework.context.annotation.Configuration class MyDataSourceConfiguration { @Bean - @ConfigurationProperties(prefix = "app.datasource") + @ConfigurationProperties("app.datasource") fun dataSource(): SomeDataSource { return SomeDataSource() } diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/howto/security/enablehttps/MySecurityConfig.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/howto/security/enablehttps/MySecurityConfig.kt index f326ca88fad5..70db2acc15a3 100644 --- a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/howto/security/enablehttps/MySecurityConfig.kt +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/howto/security/enablehttps/MySecurityConfig.kt @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,6 +18,7 @@ package org.springframework.boot.docs.howto.security.enablehttps import org.springframework.context.annotation.Bean import org.springframework.context.annotation.Configuration +import org.springframework.security.config.Customizer import org.springframework.security.config.annotation.web.builders.HttpSecurity import org.springframework.security.web.SecurityFilterChain @@ -27,7 +28,7 @@ class MySecurityConfig { @Bean fun securityFilterChain(http: HttpSecurity): SecurityFilterChain { // Customize the application security ... - http.requiresChannel { requests -> requests.anyRequest().requiresSecure() } + http.redirectToHttps { } return http.build() } diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/io/restclient/webclient/configuration/MyConnectorHttpConfiguration.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/io/restclient/webclient/configuration/MyConnectorHttpConfiguration.kt new file mode 100644 index 000000000000..0fa3385a24f1 --- /dev/null +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/io/restclient/webclient/configuration/MyConnectorHttpConfiguration.kt @@ -0,0 +1,17 @@ +package org.springframework.boot.docs.io.restclient.clienthttprequestfactory.configuration + +import org.springframework.boot.http.client.reactive.ClientHttpConnectorBuilder; +import org.springframework.context.annotation.Bean +import org.springframework.context.annotation.Configuration +import java.net.ProxySelector +import java.net.http.HttpClient + +@Configuration(proxyBeanMethods = false) +class MyConnectorHttpConfiguration { + + @Bean + fun clientHttpConnectorBuilder(proxySelector: ProxySelector): ClientHttpConnectorBuilder<*> { + return ClientHttpConnectorBuilder.jdk().withHttpClientCustomizer { builder -> builder.proxy(proxySelector) } + } + +} \ No newline at end of file diff --git a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesKotlin.kt b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesKotlin.kt index f094e0bf742a..78a9acd1ac0e 100644 --- a/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesKotlin.kt +++ b/spring-boot-project/spring-boot-docs/src/main/kotlin/org/springframework/boot/docs/packaging/nativeimage/advanced/nestedconfigurationproperties/MyPropertiesKotlin.kt @@ -19,7 +19,7 @@ package org.springframework.boot.docs.packaging.nativeimage.advanced.nestedconfi import org.springframework.boot.context.properties.ConfigurationProperties import org.springframework.boot.context.properties.NestedConfigurationProperty -@ConfigurationProperties(prefix = "my.properties") +@ConfigurationProperties("my.properties") data class MyPropertiesKotlin( val name: String, @NestedConfigurationProperty val nested: Nested diff --git a/spring-boot-project/spring-boot-parent/build.gradle b/spring-boot-project/spring-boot-parent/build.gradle index 559fc855ee0b..bafcb3104718 100644 --- a/spring-boot-project/spring-boot-parent/build.gradle +++ b/spring-boot-project/spring-boot-parent/build.gradle @@ -1,6 +1,5 @@ plugins { id "org.springframework.boot.bom" - id "org.springframework.boot.deployed" } description = "Spring Boot Parent" @@ -26,6 +25,13 @@ bom { ] } } + library("AWS Advanced JDBC Wrapper", "2.5.4") { + group("software.amazon.jdbc") { + modules = [ + "aws-advanced-jdbc-wrapper" + ] + } + } library("C3P0", "0.9.5.5") { group("com.mchange") { modules = [ @@ -55,7 +61,7 @@ bom { ] } } - library("CycloneDX Gradle Plugin", "1.10.0") { + library("CycloneDX Gradle Plugin", "2.2.0") { group("org.cyclonedx") { modules = [ "cyclonedx-gradle-plugin" @@ -64,9 +70,9 @@ bom { } library("Janino", "3.1.10") { group("org.codehaus.janino") { - imports = [ - "janino" - ] + bom("janino") { + permit("junit:junit") + } } } library("JLine", "2.11") { diff --git a/spring-boot-project/spring-boot-starters/README.adoc b/spring-boot-project/spring-boot-starters/README.adoc index f685eb1ce66b..642151980371 100644 --- a/spring-boot-project/spring-boot-starters/README.adoc +++ b/spring-boot-project/spring-boot-starters/README.adoc @@ -71,6 +71,9 @@ do as they were designed before this was clarified. | https://www.couchbase.com/[Couchbase] HTTP session | https://github.com/mkopylec/session-couchbase-spring-boot-starter +| https://dapr.io[Dapr] +| https://github.com/dapr/java-sdk/ + | DataSource decorating (https://github.com/p6spy/p6spy[P6Spy], https://github.com/ttddyy/datasource-proxy[datasource-proxy], https://github.com/vladmihalcea/flexy-pool[FlexyPool]) | https://github.com/gavlyukovskiy/spring-boot-data-source-decorator @@ -257,6 +260,9 @@ do as they were designed before this was clarified. | https://github.com/structurizr/java[Structurizr] | https://github.com/Catalysts/structurizr-extensions +| https://docs.styra.com/das/systems/springboot/[Styra DAS] (https://www.openpolicyagent.org/[OPA]) +| https://github.com/styrainc/opa-springboot + | https://www.torproject.org/[Tor] | https://github.com/theborakompanioni/tor-spring-boot-starter diff --git a/spring-boot-project/spring-boot-starters/spring-boot-starter-parent/build.gradle b/spring-boot-project/spring-boot-starters/spring-boot-starter-parent/build.gradle index f695fbfd614f..1accf86dcf18 100644 --- a/spring-boot-project/spring-boot-starters/spring-boot-starter-parent/build.gradle +++ b/spring-boot-project/spring-boot-starters/spring-boot-starter-parent/build.gradle @@ -306,31 +306,33 @@ publishing.publications.withType(MavenPublication) { } } build { - plugins { - plugin { - delegate.groupId('org.springframework.boot') - delegate.artifactId('spring-boot-maven-plugin') - executions { - execution { - delegate.id('process-test-aot') - goals { - delegate.goal('process-test-aot') + pluginManagement { + plugins { + plugin { + delegate.groupId('org.springframework.boot') + delegate.artifactId('spring-boot-maven-plugin') + executions { + execution { + delegate.id('process-test-aot') + goals { + delegate.goal('process-test-aot') + } } } } - } - plugin { - delegate.groupId('org.graalvm.buildtools') - delegate.artifactId('native-maven-plugin') - configuration { - delegate.classesDirectory('${project.build.outputDirectory}') - delegate.requiredVersion('22.3') - } - executions { - execution { - delegate.id('native-test') - goals { - delegate.goal('test') + plugin { + delegate.groupId('org.graalvm.buildtools') + delegate.artifactId('native-maven-plugin') + configuration { + delegate.classesDirectory('${project.build.outputDirectory}') + delegate.requiredVersion('22.3') + } + executions { + execution { + delegate.id('native-test') + goals { + delegate.goal('test') + } } } } diff --git a/spring-boot-project/spring-boot-test-autoconfigure/build.gradle b/spring-boot-project/spring-boot-test-autoconfigure/build.gradle index 09ab2bd8138f..4e078f780bb7 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/build.gradle +++ b/spring-boot-project/spring-boot-test-autoconfigure/build.gradle @@ -91,7 +91,7 @@ dependencies { testImplementation("com.h2database:h2") testImplementation("com.unboundid:unboundid-ldapsdk") testImplementation("io.lettuce:lettuce-core") - testImplementation("io.micrometer:micrometer-registry-prometheus-simpleclient") + testImplementation("io.micrometer:micrometer-registry-prometheus") testImplementation("io.projectreactor.netty:reactor-netty-http") testImplementation("io.projectreactor:reactor-core") testImplementation("io.projectreactor:reactor-test") diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/ConditionReportApplicationContextFailureProcessor.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/ConditionReportApplicationContextFailureProcessor.java index 66e45638b8b4..1735c7671d36 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/ConditionReportApplicationContextFailureProcessor.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/ConditionReportApplicationContextFailureProcessor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Phillip Webb * @author Scott Frederick * @since 3.0.0 - * @deprecated in 3.2.11 for removal in 3.6.0 + * @deprecated in 3.2.11 for removal in 4.0.0 */ @Deprecated(since = "3.2.11", forRemoval = true) public class ConditionReportApplicationContextFailureProcessor implements ApplicationContextFailureProcessor { diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/filter/FilterAnnotations.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/filter/FilterAnnotations.java index 8626043851c8..d9a1bd2d4dc3 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/filter/FilterAnnotations.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/filter/FilterAnnotations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,7 @@ public class FilterAnnotations implements Iterable { private final List filters; public FilterAnnotations(ClassLoader classLoader, Filter[] filters) { - Assert.notNull(filters, "Filters must not be null"); + Assert.notNull(filters, "'filters' must not be null"); this.classLoader = classLoader; this.filters = createTypeFilters(filters); } @@ -72,16 +72,16 @@ private TypeFilter createTypeFilter(FilterType filterType, Class filterClass) return switch (filterType) { case ANNOTATION -> { Assert.isAssignable(Annotation.class, filterClass, - "An error occurred while processing an ANNOTATION type filter: "); + "'filterClass' must be an Annotation when 'filterType' is ANNOTATION"); yield new AnnotationTypeFilter((Class) filterClass); } case ASSIGNABLE_TYPE -> new AssignableTypeFilter(filterClass); case CUSTOM -> { Assert.isAssignable(TypeFilter.class, filterClass, - "An error occurred while processing a CUSTOM type filter: "); + "'filterClass' must be a TypeFilter when 'filterType' is CUSTOM"); yield BeanUtils.instantiateClass(filterClass, TypeFilter.class); } - default -> throw new IllegalArgumentException("Filter type not supported with Class value: " + filterType); + default -> throw new IllegalArgumentException("'filterClass' not supported [" + filterType + "]"); }; } diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/graphql/tester/HttpGraphQlTesterAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/graphql/tester/HttpGraphQlTesterAutoConfiguration.java index 953571a1c9bc..2523badfa359 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/graphql/tester/HttpGraphQlTesterAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/graphql/tester/HttpGraphQlTesterAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,7 +43,7 @@ public class HttpGraphQlTesterAutoConfiguration { @ConditionalOnBean(WebTestClient.class) @ConditionalOnMissingBean public HttpGraphQlTester webTestClientGraphQlTester(WebTestClient webTestClient, GraphQlProperties properties) { - WebTestClient mutatedWebTestClient = webTestClient.mutate().baseUrl(properties.getPath()).build(); + WebTestClient mutatedWebTestClient = webTestClient.mutate().baseUrl(properties.getHttp().getPath()).build(); return HttpGraphQlTester.create(mutatedWebTestClient); } diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/jdbc/TestDatabaseAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/jdbc/TestDatabaseAutoConfiguration.java index cc59cbf970a4..10b099f61edb 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/jdbc/TestDatabaseAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/jdbc/TestDatabaseAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -79,8 +79,7 @@ public class TestDatabaseAutoConfiguration { @Bean @Role(BeanDefinition.ROLE_INFRASTRUCTURE) - @ConditionalOnProperty(prefix = "spring.test.database", name = "replace", havingValue = "NON_TEST", - matchIfMissing = true) + @ConditionalOnProperty(name = "spring.test.database.replace", havingValue = "NON_TEST", matchIfMissing = true) static EmbeddedDataSourceBeanFactoryPostProcessor nonTestEmbeddedDataSourceBeanFactoryPostProcessor( Environment environment) { return new EmbeddedDataSourceBeanFactoryPostProcessor(environment, Replace.NON_TEST); @@ -88,14 +87,14 @@ static EmbeddedDataSourceBeanFactoryPostProcessor nonTestEmbeddedDataSourceBeanF @Bean @Role(BeanDefinition.ROLE_INFRASTRUCTURE) - @ConditionalOnProperty(prefix = "spring.test.database", name = "replace", havingValue = "ANY") + @ConditionalOnProperty(name = "spring.test.database.replace", havingValue = "ANY") static EmbeddedDataSourceBeanFactoryPostProcessor embeddedDataSourceBeanFactoryPostProcessor( Environment environment) { return new EmbeddedDataSourceBeanFactoryPostProcessor(environment, Replace.ANY); } @Bean - @ConditionalOnProperty(prefix = "spring.test.database", name = "replace", havingValue = "AUTO_CONFIGURED") + @ConditionalOnProperty(name = "spring.test.database.replace", havingValue = "AUTO_CONFIGURED") @ConditionalOnMissingBean public DataSource dataSource(Environment environment) { return new EmbeddedDataSourceFactory(environment).getEmbeddedDatabase(); @@ -127,8 +126,8 @@ public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) t if (AotDetector.useGeneratedArtifacts()) { return; } - Assert.isInstanceOf(ConfigurableListableBeanFactory.class, registry, - "Test Database Auto-configuration can only be used with a ConfigurableListableBeanFactory"); + Assert.isTrue(registry instanceof ConfigurableListableBeanFactory, + "'registry' must be a ConfigurableListableBeanFactory"); process(registry, (ConfigurableListableBeanFactory) registry); } diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/json/JsonTestersAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/json/JsonTestersAutoConfiguration.java index a5d223820afb..818ff6cf1aa8 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/json/JsonTestersAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/json/JsonTestersAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,8 +36,8 @@ import org.springframework.beans.factory.config.InstantiationAwareBeanPostProcessor; import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.gson.GsonAutoConfiguration; import org.springframework.boot.autoconfigure.jackson.JacksonAutoConfiguration; import org.springframework.boot.autoconfigure.jsonb.JsonbAutoConfiguration; @@ -65,7 +65,7 @@ @AutoConfiguration( after = { JacksonAutoConfiguration.class, GsonAutoConfiguration.class, JsonbAutoConfiguration.class }) @ConditionalOnClass(name = "org.assertj.core.api.Assert") -@ConditionalOnProperty("spring.test.jsontesters.enabled") +@ConditionalOnBooleanProperty("spring.test.jsontesters.enabled") public class JsonTestersAutoConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/orm/jpa/TestEntityManager.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/orm/jpa/TestEntityManager.java index bd9c22d2203a..3e24d6579dae 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/orm/jpa/TestEntityManager.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/orm/jpa/TestEntityManager.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,7 @@ public class TestEntityManager { * @param entityManagerFactory the source entity manager factory */ public TestEntityManager(EntityManagerFactory entityManagerFactory) { - Assert.notNull(entityManagerFactory, "EntityManagerFactory must not be null"); + Assert.notNull(entityManagerFactory, "'entityManagerFactory' must not be null"); this.entityManagerFactory = entityManagerFactory; } diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/client/MockRestServiceServerAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/client/MockRestServiceServerAutoConfiguration.java index a0ada9a512c4..9743088e10cb 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/client/MockRestServiceServerAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/client/MockRestServiceServerAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,7 +23,7 @@ import java.util.Map; import org.springframework.boot.autoconfigure.AutoConfiguration; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.test.web.client.MockServerRestClientCustomizer; import org.springframework.boot.test.web.client.MockServerRestTemplateCustomizer; import org.springframework.context.annotation.Bean; @@ -47,7 +47,7 @@ * @see AutoConfigureMockRestServiceServer */ @AutoConfiguration -@ConditionalOnProperty(prefix = "spring.test.webclient.mockrestserviceserver", name = "enabled") +@ConditionalOnBooleanProperty("spring.test.webclient.mockrestserviceserver.enabled") public class MockRestServiceServerAutoConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/client/WebClientRestTemplateAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/client/WebClientRestTemplateAutoConfiguration.java index 108e3abbb7df..0c6666354f85 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/client/WebClientRestTemplateAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/client/WebClientRestTemplateAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,7 +17,7 @@ package org.springframework.boot.test.autoconfigure.web.client; import org.springframework.boot.autoconfigure.AutoConfiguration; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.web.client.RestTemplateAutoConfiguration; import org.springframework.boot.web.client.RestTemplateBuilder; import org.springframework.context.annotation.Bean; @@ -32,7 +32,7 @@ * @see AutoConfigureMockRestServiceServer */ @AutoConfiguration(after = RestTemplateAutoConfiguration.class) -@ConditionalOnProperty(prefix = "spring.test.webclient", name = "register-rest-template") +@ConditionalOnBooleanProperty("spring.test.webclient.register-rest-template") public class WebClientRestTemplateAutoConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/reactive/WebTestClientAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/reactive/WebTestClientAutoConfiguration.java index 279397ceed01..28962402fae4 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/reactive/WebTestClientAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/reactive/WebTestClientAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -67,7 +67,7 @@ public WebTestClient webTestClient(ApplicationContext applicationContext, } @Bean - @ConfigurationProperties(prefix = "spring.test.webtestclient") + @ConfigurationProperties("spring.test.webtestclient") public SpringBootWebTestClientBuilderCustomizer springBootWebTestClientBuilderCustomizer( ObjectProvider codecCustomizers) { return new SpringBootWebTestClientBuilderCustomizer(codecCustomizers.orderedStream().toList()); diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcConfiguration.java index 2ee2b145cc61..7428917e1e99 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -60,7 +60,7 @@ DefaultMockMvcBuilder mockMvcBuilder(List customizers) } @Bean - @ConfigurationProperties(prefix = "spring.test.mockmvc") + @ConfigurationProperties("spring.test.mockmvc") SpringBootMockMvcBuilderCustomizer springBootMockMvcBuilderCustomizer() { return new SpringBootMockMvcBuilderCustomizer(this.context); } diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcWebClientAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcWebClientAutoConfiguration.java index 9f838bb2c4ca..1fe329cc35f6 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcWebClientAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcWebClientAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,9 +20,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.test.web.htmlunit.LocalHostWebClient; import org.springframework.context.annotation.Bean; import org.springframework.core.env.Environment; @@ -37,7 +37,7 @@ */ @AutoConfiguration(after = MockMvcAutoConfiguration.class) @ConditionalOnClass(WebClient.class) -@ConditionalOnProperty(prefix = "spring.test.mockmvc.webclient", name = "enabled", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.test.mockmvc.webclient.enabled", matchIfMissing = true) public class MockMvcWebClientAutoConfiguration { @Bean diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcWebDriverAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcWebDriverAutoConfiguration.java index 6a45c59c7b21..92c3bb46f0c8 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcWebDriverAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/MockMvcWebDriverAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,9 +24,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.test.web.htmlunit.webdriver.LocalHostWebConnectionHtmlUnitDriver; import org.springframework.context.annotation.Bean; import org.springframework.core.env.Environment; @@ -43,7 +43,7 @@ */ @AutoConfiguration(after = MockMvcAutoConfiguration.class) @ConditionalOnClass(HtmlUnitDriver.class) -@ConditionalOnProperty(prefix = "spring.test.mockmvc.webdriver", name = "enabled", matchIfMissing = true) +@ConditionalOnBooleanProperty(name = "spring.test.mockmvc.webdriver.enabled", matchIfMissing = true) public class MockMvcWebDriverAutoConfiguration { private static final String SECURITY_CONTEXT_EXECUTOR = "org.springframework.security.concurrent.DelegatingSecurityContextExecutor"; diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/SpringBootMockMvcBuilderCustomizer.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/SpringBootMockMvcBuilderCustomizer.java index 693a5431d03c..6df4c1338926 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/SpringBootMockMvcBuilderCustomizer.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/web/servlet/SpringBootMockMvcBuilderCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -70,7 +70,7 @@ public class SpringBootMockMvcBuilderCustomizer implements MockMvcBuilderCustomi * @param context the source application context */ public SpringBootMockMvcBuilderCustomizer(WebApplicationContext context) { - Assert.notNull(context, "Context must not be null"); + Assert.notNull(context, "'context' must not be null"); this.context = context; } diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/webservices/client/MockWebServiceServerAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/webservices/client/MockWebServiceServerAutoConfiguration.java index 87b767ee8bf7..940e77b994f8 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/webservices/client/MockWebServiceServerAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/webservices/client/MockWebServiceServerAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,8 +17,8 @@ package org.springframework.boot.test.autoconfigure.webservices.client; import org.springframework.boot.autoconfigure.AutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Bean; import org.springframework.ws.client.core.WebServiceTemplate; import org.springframework.ws.test.client.MockWebServiceMessageSender; @@ -32,7 +32,7 @@ * @see AutoConfigureMockWebServiceServer */ @AutoConfiguration -@ConditionalOnProperty(prefix = "spring.test.webservice.client.mockserver", name = "enabled") +@ConditionalOnBooleanProperty("spring.test.webservice.client.mockserver.enabled") @ConditionalOnClass({ MockWebServiceServer.class, WebServiceTemplate.class }) public class MockWebServiceServerAutoConfiguration { diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/webservices/client/WebServiceClientTemplateAutoConfiguration.java b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/webservices/client/WebServiceClientTemplateAutoConfiguration.java index 96587af44615..7f69059984b1 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/webservices/client/WebServiceClientTemplateAutoConfiguration.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/main/java/org/springframework/boot/test/autoconfigure/webservices/client/WebServiceClientTemplateAutoConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -18,8 +18,8 @@ import org.springframework.boot.autoconfigure.AutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnBooleanProperty; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; -import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.webservices.client.WebServiceTemplateAutoConfiguration; import org.springframework.boot.webservices.client.WebServiceTemplateBuilder; import org.springframework.context.annotation.Bean; @@ -34,7 +34,7 @@ * @see AutoConfigureWebServiceClient */ @AutoConfiguration(after = WebServiceTemplateAutoConfiguration.class) -@ConditionalOnProperty(prefix = "spring.test.webservice.client", name = "register-web-service-template") +@ConditionalOnBooleanProperty("spring.test.webservice.client.register-web-service-template") @ConditionalOnClass(WebServiceTemplate.class) @ConditionalOnBean(WebServiceTemplateBuilder.class) public class WebServiceClientTemplateAutoConfiguration { diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/ConditionReportApplicationContextFailureProcessorTests.java b/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/ConditionReportApplicationContextFailureProcessorTests.java index 21b7bda6d803..1c0f1099cf03 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/ConditionReportApplicationContextFailureProcessorTests.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/ConditionReportApplicationContextFailureProcessorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ * * @author Phillip Webb * @author Scott Frederick - * @deprecated since 3.2.11 for removal in 3.6.0 + * @deprecated since 3.2.11 for removal in 4.0.0 */ @ExtendWith(OutputCaptureExtension.class) @Deprecated(since = "3.2.11", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/actuate/observability/AutoConfigureObservabilityMissingIntegrationTests.java b/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/actuate/observability/AutoConfigureObservabilityMissingIntegrationTests.java index 8c3fd64eec95..a8224996c1b8 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/actuate/observability/AutoConfigureObservabilityMissingIntegrationTests.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/actuate/observability/AutoConfigureObservabilityMissingIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,14 +36,14 @@ * @author Moritz Halbritter */ @SpringBootTest -@SuppressWarnings("deprecation") class AutoConfigureObservabilityMissingIntegrationTests { @Test void customizerRunsAndOnlyEnablesSimpleMeterRegistryWhenNoAnnotationPresent( @Autowired ApplicationContext applicationContext) { assertThat(applicationContext.getBean(MeterRegistry.class)).isInstanceOf(SimpleMeterRegistry.class); - assertThat(applicationContext.getBeansOfType(io.micrometer.prometheus.PrometheusMeterRegistry.class)).isEmpty(); + assertThat(applicationContext.getBeansOfType(io.micrometer.prometheusmetrics.PrometheusMeterRegistry.class)) + .isEmpty(); } @Test diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/actuate/observability/AutoConfigureObservabilityPresentIntegrationTests.java b/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/actuate/observability/AutoConfigureObservabilityPresentIntegrationTests.java index f1c58916c458..8fb179f6c04f 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/actuate/observability/AutoConfigureObservabilityPresentIntegrationTests.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/actuate/observability/AutoConfigureObservabilityPresentIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,13 +35,12 @@ */ @SpringBootTest @AutoConfigureObservability -@SuppressWarnings("deprecation") class AutoConfigureObservabilityPresentIntegrationTests { @Test void customizerDoesNotDisableAvailableMeterRegistriesWhenAnnotationPresent( @Autowired ApplicationContext applicationContext) { - assertThat(applicationContext.getBeansOfType(io.micrometer.prometheus.PrometheusMeterRegistry.class)) + assertThat(applicationContext.getBeansOfType(io.micrometer.prometheusmetrics.PrometheusMeterRegistry.class)) .hasSize(1); } diff --git a/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/orm/jpa/TestEntityManagerTests.java b/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/orm/jpa/TestEntityManagerTests.java index 8ec9c28b7d28..fb8e9a749ff0 100644 --- a/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/orm/jpa/TestEntityManagerTests.java +++ b/spring-boot-project/spring-boot-test-autoconfigure/src/test/java/org/springframework/boot/test/autoconfigure/orm/jpa/TestEntityManagerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -61,7 +61,7 @@ void setup() { @Test void createWhenEntityManagerIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new TestEntityManager(null)) - .withMessageContaining("EntityManagerFactory must not be null"); + .withMessageContaining("'entityManagerFactory' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/AnnotatedClassFinder.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/AnnotatedClassFinder.java index 9fdf7803b203..e0cc152ac4b6 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/AnnotatedClassFinder.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/AnnotatedClassFinder.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,7 @@ public final class AnnotatedClassFinder { * @param annotationType the annotation to find */ public AnnotatedClassFinder(Class annotationType) { - Assert.notNull(annotationType, "AnnotationType must not be null"); + Assert.notNull(annotationType, "'annotationType' must not be null"); this.annotationType = annotationType; this.scanner = new ClassPathScanningCandidateComponentProvider(false); this.scanner.addIncludeFilter(new AnnotationTypeFilter(annotationType)); @@ -64,7 +64,7 @@ public AnnotatedClassFinder(Class annotationType) { * hierarchy defined by the given {@code source} or {@code null} if none is found. */ public Class findFromClass(Class source) { - Assert.notNull(source, "Source must not be null"); + Assert.notNull(source, "'source' must not be null"); return findFromPackage(ClassUtils.getPackageName(source)); } @@ -76,7 +76,7 @@ public Class findFromClass(Class source) { * hierarchy defined by the given {@code source} or {@code null} if none is found. */ public Class findFromPackage(String source) { - Assert.notNull(source, "Source must not be null"); + Assert.notNull(source, "'source' must not be null"); Class configuration = cache.get(source); if (configuration == null) { configuration = scanPackage(source); diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/assertj/ApplicationContextAssert.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/assertj/ApplicationContextAssert.java index 02fa5109d8bf..44d0dc491f76 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/assertj/ApplicationContextAssert.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/assertj/ApplicationContextAssert.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -63,7 +63,7 @@ public class ApplicationContextAssert */ ApplicationContextAssert(C applicationContext, Throwable startupFailure) { super(applicationContext, ApplicationContextAssert.class); - Assert.notNull(applicationContext, "ApplicationContext must not be null"); + Assert.notNull(applicationContext, "'applicationContext' must not be null"); this.startupFailure = startupFailure; } @@ -121,7 +121,7 @@ public ApplicationContextAssert hasSingleBean(Class type) { * given type */ public ApplicationContextAssert hasSingleBean(Class type, Scope scope) { - Assert.notNull(scope, "Scope must not be null"); + Assert.notNull(scope, "'scope' must not be null"); if (this.startupFailure != null) { throwAssertionError(contextFailedToStartWhenExpecting("to have a single bean of type:%n <%s>", type)); } @@ -168,7 +168,7 @@ public ApplicationContextAssert doesNotHaveBean(Class type) { * type */ public ApplicationContextAssert doesNotHaveBean(Class type, Scope scope) { - Assert.notNull(scope, "Scope must not be null"); + Assert.notNull(scope, "'scope' must not be null"); if (this.startupFailure != null) { throwAssertionError(contextFailedToStartWhenExpecting("not to have any beans of type:%n <%s>", type)); } @@ -265,7 +265,7 @@ public AbstractObjectAssert getBean(Class type) { * given type */ public AbstractObjectAssert getBean(Class type, Scope scope) { - Assert.notNull(scope, "Scope must not be null"); + Assert.notNull(scope, "'scope' must not be null"); if (this.startupFailure != null) { throwAssertionError(contextFailedToStartWhenExpecting("to contain bean of type:%n <%s>", type)); } @@ -407,7 +407,7 @@ public MapAssert getBeans(Class type) { * @throws AssertionError if the application context did not start */ public MapAssert getBeans(Class type, Scope scope) { - Assert.notNull(scope, "Scope must not be null"); + Assert.notNull(scope, "'scope' must not be null"); if (this.startupFailure != null) { throwAssertionError(contextFailedToStartWhenExpecting("to get beans of type:%n <%s>", type)); } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertProvider.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertProvider.java index 8542c5aa675c..f6219e83efba 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertProvider.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertProvider.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -127,10 +127,10 @@ static , C extends ApplicationCont static , C extends ApplicationContext> T get(Class type, Class contextType, Supplier contextSupplier, Class... additionalContextInterfaces) { - Assert.notNull(type, "Type must not be null"); - Assert.isTrue(type.isInterface(), "Type must be an interface"); - Assert.notNull(contextType, "ContextType must not be null"); - Assert.isTrue(contextType.isInterface(), "ContextType must be an interface"); + Assert.notNull(type, "'type' must not be null"); + Assert.isTrue(type.isInterface(), "'type' must be an interface"); + Assert.notNull(contextType, "'contextType' must not be null"); + Assert.isTrue(contextType.isInterface(), "'contextType' must be an interface"); Class[] interfaces = merge(new Class[] { type, contextType }, additionalContextInterfaces); return (T) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(), interfaces, new AssertProviderApplicationContextInvocationHandler(contextType, contextSupplier)); diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/runner/AbstractApplicationContextRunner.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/runner/AbstractApplicationContextRunner.java index 5bb815e9a48c..26c7d022a62a 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/runner/AbstractApplicationContextRunner.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/runner/AbstractApplicationContextRunner.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -121,7 +121,7 @@ public abstract class AbstractApplicationContextRunner, SELF */ protected AbstractApplicationContextRunner(RunnerConfiguration configuration, Function, SELF> instanceFactory) { - Assert.notNull(configuration, "RunnerConfiguration must not be null"); - Assert.notNull(instanceFactory, "instanceFactory must not be null"); + Assert.notNull(configuration, "'configuration' must not be null"); + Assert.notNull(instanceFactory, "'instanceFactory' must not be null"); this.runnerConfiguration = configuration; this.instanceFactory = instanceFactory; } @@ -190,7 +190,7 @@ public SELF withAllowCircularReferences(boolean allowCircularReferences) { * @return a new instance with the updated initializers */ public SELF withInitializer(ApplicationContextInitializer initializer) { - Assert.notNull(initializer, "Initializer must not be null"); + Assert.notNull(initializer, "'initializer' must not be null"); return newInstance(this.runnerConfiguration.withInitializer(initializer)); } @@ -332,7 +332,7 @@ public SELF withUserConfiguration(Class... configurationClasses) { * @return a new instance with the updated configuration */ public SELF withConfiguration(Configurations configurations) { - Assert.notNull(configurations, "Configurations must not be null"); + Assert.notNull(configurations, "'configurations' must not be null"); return newInstance(this.runnerConfiguration.withConfiguration(configurations)); } @@ -559,7 +559,7 @@ private RunnerConfiguration withAllowCircularReferences(boolean allowCircular } private RunnerConfiguration withInitializer(ApplicationContextInitializer initializer) { - Assert.notNull(initializer, "Initializer must not be null"); + Assert.notNull(initializer, "'initializer' must not be null"); RunnerConfiguration config = new RunnerConfiguration<>(this); config.initializers = add(config.initializers, initializer); return config; @@ -605,7 +605,7 @@ private RunnerConfiguration withBean(String name, Class type, Supplier } private RunnerConfiguration withConfiguration(Configurations configurations) { - Assert.notNull(configurations, "Configurations must not be null"); + Assert.notNull(configurations, "'configurations' must not be null"); RunnerConfiguration config = new RunnerConfiguration<>(this); config.configurations = add(config.configurations, configurations); return config; diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/runner/ContextConsumer.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/runner/ContextConsumer.java index 8c80978726c8..c11829effa1c 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/runner/ContextConsumer.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/context/runner/ContextConsumer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,7 +48,7 @@ public interface ContextConsumer { * @since 2.6.0 */ default ContextConsumer andThen(ContextConsumer after) { - Assert.notNull(after, "After must not be null"); + Assert.notNull(after, "'after' must not be null"); return (context) -> { accept(context); after.accept(context); diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/graphql/tester/HttpGraphQlTesterContextCustomizer.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/graphql/tester/HttpGraphQlTesterContextCustomizer.java index 446063bd8c06..ba3c53ed98b4 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/graphql/tester/HttpGraphQlTesterContextCustomizer.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/graphql/tester/HttpGraphQlTesterContextCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2020-2024 the original author or authors. + * Copyright 2020-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -171,7 +171,7 @@ private String deduceBasePath() { } private String findConfiguredGraphQlPath() { - String configuredPath = this.applicationContext.getEnvironment().getProperty("spring.graphql.path"); + String configuredPath = this.applicationContext.getEnvironment().getProperty("spring.graphql.http.path"); return StringUtils.hasText(configuredPath) ? configuredPath : "/graphql"; } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/AbstractJsonMarshalTester.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/AbstractJsonMarshalTester.java index ae9066273f0e..4eb2f11204e3 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/AbstractJsonMarshalTester.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/AbstractJsonMarshalTester.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -85,8 +85,8 @@ protected AbstractJsonMarshalTester() { * @param type the type under test */ public AbstractJsonMarshalTester(Class resourceLoadClass, ResolvableType type) { - Assert.notNull(resourceLoadClass, "ResourceLoadClass must not be null"); - Assert.notNull(type, "Type must not be null"); + Assert.notNull(resourceLoadClass, "'resourceLoadClass' must not be null"); + Assert.notNull(type, "'type' must not be null"); initialize(resourceLoadClass, type); } @@ -127,7 +127,7 @@ protected final Class getResourceLoadClass() { */ public JsonContent write(T value) throws IOException { verify(); - Assert.notNull(value, "Value must not be null"); + Assert.notNull(value, "'value' must not be null"); String json = writeObject(value, this.type); return getJsonContent(json); } @@ -162,7 +162,7 @@ public T parseObject(byte[] jsonBytes) throws IOException { */ public ObjectContent parse(byte[] jsonBytes) throws IOException { verify(); - Assert.notNull(jsonBytes, "JsonBytes must not be null"); + Assert.notNull(jsonBytes, "'jsonBytes' must not be null"); return read(new ByteArrayResource(jsonBytes)); } @@ -185,7 +185,7 @@ public T parseObject(String jsonString) throws IOException { */ public ObjectContent parse(String jsonString) throws IOException { verify(); - Assert.notNull(jsonString, "JsonString must not be null"); + Assert.notNull(jsonString, "'jsonString' must not be null"); return read(new StringReader(jsonString)); } @@ -210,7 +210,7 @@ public T readObject(String resourcePath) throws IOException { */ public ObjectContent read(String resourcePath) throws IOException { verify(); - Assert.notNull(resourcePath, "ResourcePath must not be null"); + Assert.notNull(resourcePath, "'resourcePath' must not be null"); return read(new ClassPathResource(resourcePath, this.resourceLoadClass)); } @@ -233,7 +233,7 @@ public T readObject(File file) throws IOException { */ public ObjectContent read(File file) throws IOException { verify(); - Assert.notNull(file, "File must not be null"); + Assert.notNull(file, "'file' must not be null"); return read(new FileSystemResource(file)); } @@ -256,7 +256,7 @@ public T readObject(InputStream inputStream) throws IOException { */ public ObjectContent read(InputStream inputStream) throws IOException { verify(); - Assert.notNull(inputStream, "InputStream must not be null"); + Assert.notNull(inputStream, "'inputStream' must not be null"); return read(new InputStreamResource(inputStream)); } @@ -279,7 +279,7 @@ public T readObject(Resource resource) throws IOException { */ public ObjectContent read(Resource resource) throws IOException { verify(); - Assert.notNull(resource, "Resource must not be null"); + Assert.notNull(resource, "'resource' must not be null"); InputStream inputStream = resource.getInputStream(); T object = readObject(inputStream, this.type); closeQuietly(inputStream); @@ -305,7 +305,7 @@ public T readObject(Reader reader) throws IOException { */ public ObjectContent read(Reader reader) throws IOException { verify(); - Assert.notNull(reader, "Reader must not be null"); + Assert.notNull(reader, "'reader' must not be null"); T object = readObject(reader, this.type); closeQuietly(reader); return new ObjectContent<>(this.type, object); @@ -368,19 +368,19 @@ protected abstract static class FieldInitializer { @SuppressWarnings("rawtypes") protected FieldInitializer(Class testerClass) { - Assert.notNull(testerClass, "TesterClass must not be null"); + Assert.notNull(testerClass, "'testerClass' must not be null"); this.testerClass = testerClass; } public void initFields(Object testInstance, M marshaller) { - Assert.notNull(testInstance, "TestInstance must not be null"); - Assert.notNull(marshaller, "Marshaller must not be null"); + Assert.notNull(testInstance, "'testInstance' must not be null"); + Assert.notNull(marshaller, "'marshaller' must not be null"); initFields(testInstance, () -> marshaller); } public void initFields(Object testInstance, final ObjectFactory marshaller) { - Assert.notNull(testInstance, "TestInstance must not be null"); - Assert.notNull(marshaller, "Marshaller must not be null"); + Assert.notNull(testInstance, "'testInstance' must not be null"); + Assert.notNull(marshaller, "'marshaller' must not be null"); ReflectionUtils.doWithFields(testInstance.getClass(), (field) -> doWithField(field, testInstance, marshaller)); } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/BasicJsonTester.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/BasicJsonTester.java index 9b9bf84b3af1..71f6dea0b3f0 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/BasicJsonTester.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/BasicJsonTester.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -70,7 +70,7 @@ public BasicJsonTester(Class resourceLoadClass) { * @since 1.4.1 */ public BasicJsonTester(Class resourceLoadClass, Charset charset) { - Assert.notNull(resourceLoadClass, "ResourceLoadClass must not be null"); + Assert.notNull(resourceLoadClass, "'resourceLoadClass' must not be null"); this.loader = new JsonLoader(resourceLoadClass, charset); } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/GsonTester.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/GsonTester.java index 797c98918c88..d8f7522eb82f 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/GsonTester.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/GsonTester.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -62,7 +62,7 @@ public class GsonTester extends AbstractJsonMarshalTester { * @param gson the Gson instance */ protected GsonTester(Gson gson) { - Assert.notNull(gson, "Gson must not be null"); + Assert.notNull(gson, "'gson' must not be null"); this.gson = gson; } @@ -75,7 +75,7 @@ protected GsonTester(Gson gson) { */ public GsonTester(Class resourceLoadClass, ResolvableType type, Gson gson) { super(resourceLoadClass, type); - Assert.notNull(gson, "Gson must not be null"); + Assert.notNull(gson, "'gson' must not be null"); this.gson = gson; } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JacksonTester.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JacksonTester.java index 97d58ff8ec42..3fb169b4ca8b 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JacksonTester.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JacksonTester.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -73,7 +73,7 @@ public class JacksonTester extends AbstractJsonMarshalTester { * @param objectMapper the Jackson object mapper */ protected JacksonTester(ObjectMapper objectMapper) { - Assert.notNull(objectMapper, "ObjectMapper must not be null"); + Assert.notNull(objectMapper, "'objectMapper' must not be null"); this.objectMapper = objectMapper; } @@ -89,7 +89,7 @@ public JacksonTester(Class resourceLoadClass, ResolvableType type, ObjectMapp public JacksonTester(Class resourceLoadClass, ResolvableType type, ObjectMapper objectMapper, Class view) { super(resourceLoadClass, type); - Assert.notNull(objectMapper, "ObjectMapper must not be null"); + Assert.notNull(objectMapper, "'objectMapper' must not be null"); this.objectMapper = objectMapper; this.view = view; } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonContent.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonContent.java index 94169a62e854..98ff2675af6c 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonContent.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonContent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -60,9 +60,9 @@ public JsonContent(Class resourceLoadClass, ResolvableType type, String json) * @param configuration the JsonPath configuration */ JsonContent(Class resourceLoadClass, ResolvableType type, String json, Configuration configuration) { - Assert.notNull(resourceLoadClass, "ResourceLoadClass must not be null"); - Assert.notNull(json, "JSON must not be null"); - Assert.notNull(configuration, "Configuration must not be null"); + Assert.notNull(resourceLoadClass, "'resourceLoadClass' must not be null"); + Assert.notNull(json, "'json' must not be null"); + Assert.notNull(configuration, "'configuration' must not be null"); this.resourceLoadClass = resourceLoadClass; this.type = type; this.json = json; diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonContentAssert.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonContentAssert.java index 797abc24de5f..17d6c5ccfb2b 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonContentAssert.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonContentAssert.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -1060,7 +1060,7 @@ private class JsonPathValue { JsonPathValue(CharSequence expression, Object... args) { org.springframework.util.Assert.hasText((expression != null) ? expression.toString() : null, - "expression must not be null or empty"); + "'expression' must not be empty"); this.expression = String.format(expression.toString(), args); this.jsonPath = JsonPath.compile(this.expression); } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonbTester.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonbTester.java index 988e5d30329f..4a6a56d67f0a 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonbTester.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/JsonbTester.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -62,7 +62,7 @@ public class JsonbTester extends AbstractJsonMarshalTester { * @param jsonb the Jsonb instance */ protected JsonbTester(Jsonb jsonb) { - Assert.notNull(jsonb, "Jsonb must not be null"); + Assert.notNull(jsonb, "'jsonb' must not be null"); this.jsonb = jsonb; } @@ -75,7 +75,7 @@ protected JsonbTester(Jsonb jsonb) { */ public JsonbTester(Class resourceLoadClass, ResolvableType type, Jsonb jsonb) { super(resourceLoadClass, type); - Assert.notNull(jsonb, "Jsonb must not be null"); + Assert.notNull(jsonb, "'jsonb' must not be null"); this.jsonb = jsonb; } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/ObjectContent.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/ObjectContent.java index 8535b82b0198..85e9e0753907 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/ObjectContent.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/json/ObjectContent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,7 +42,7 @@ public final class ObjectContent implements AssertProvider[] extraInterfaces, Answers answer, boolean serializable, MockReset reset, QualifierDefinition qualifier) { super(name, reset, false, qualifier); - Assert.notNull(typeToMock, "TypeToMock must not be null"); + Assert.notNull(typeToMock, "'typeToMock' must not be null"); this.typeToMock = typeToMock; this.extraInterfaces = asClassSet(extraInterfaces); this.answer = (answer != null) ? answer : Answers.RETURNS_DEFAULTS; diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockReset.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockReset.java index d4e77598801e..aa6894de775c 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockReset.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockReset.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,10 +35,9 @@ * @author Phillip Webb * @since 1.4.0 * @see ResetMocksTestExecutionListener - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link org.springframework.test.context.bean.override.mockito.MockReset} */ - @Deprecated(since = "3.4.0", forRemoval = true) public enum MockReset { @@ -92,7 +91,7 @@ public static MockSettings withSettings(MockReset reset) { * @return the configured settings */ public static MockSettings apply(MockReset reset, MockSettings settings) { - Assert.notNull(settings, "Settings must not be null"); + Assert.notNull(settings, "'settings' must not be null"); if (reset != null && reset != NONE) { settings.invocationListeners(new ResetInvocationListener(reset)); } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizer.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizer.java index ad6f70474a55..e28bdc496288 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizer.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizer.java @@ -28,7 +28,7 @@ * A {@link ContextCustomizer} to add Mockito support. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerFactory.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerFactory.java index 9946ddfb1be2..44f993d8aaa9 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerFactory.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerFactory.java @@ -27,7 +27,7 @@ * A {@link ContextCustomizerFactory} to add Mockito support. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoPostProcessor.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoPostProcessor.java index cb5774d092d7..6218330b7461 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoPostProcessor.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoPostProcessor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -78,7 +78,7 @@ * @author Stephane Nicoll * @author Andreas Neiser * @since 1.4.0 - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of Spring Framework's + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of Spring Framework's * {@link MockitoBean} and {@link MockitoSpyBean} support */ @SuppressWarnings("removal") @@ -123,15 +123,14 @@ public void setBeanClassLoader(ClassLoader classLoader) { @Override public void setBeanFactory(BeanFactory beanFactory) throws BeansException { - Assert.isInstanceOf(ConfigurableListableBeanFactory.class, beanFactory, - "Mock beans can only be used with a ConfigurableListableBeanFactory"); + Assert.isTrue(beanFactory instanceof ConfigurableListableBeanFactory, + "'beanFactory' must be a ConfigurableListableBeanFactory"); this.beanFactory = beanFactory; } @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { - Assert.isInstanceOf(BeanDefinitionRegistry.class, beanFactory, - "@MockBean can only be used on bean factories that implement BeanDefinitionRegistry"); + Assert.isTrue(beanFactory instanceof BeanDefinitionRegistry, "'beanFactory' must be a BeanDefinitionRegistry"); postProcessBeanFactory(beanFactory, (BeanDefinitionRegistry) beanFactory); } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListener.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListener.java index 941b8784031f..e4d72905d15f 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListener.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListener.java @@ -48,7 +48,7 @@ * @author Moritz Halbritter * @since 1.4.2 * @see ResetMocksTestExecutionListener - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of Spring Framework's support for + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of Spring Framework's support for * {@link MockitoBean} and {@link MockitoSpyBean}. */ @SuppressWarnings("removal") diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/QualifierDefinition.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/QualifierDefinition.java index d077a2af1ed8..3139311cb5ee 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/QualifierDefinition.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/QualifierDefinition.java @@ -34,7 +34,7 @@ * @author Phillip Webb * @author Stephane Nicoll * @see Definition - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/ResetMocksTestExecutionListener.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/ResetMocksTestExecutionListener.java index 7126b0d4dc46..5c569971957e 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/ResetMocksTestExecutionListener.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/ResetMocksTestExecutionListener.java @@ -43,7 +43,7 @@ * @author Phillip Webb * @since 1.4.0 * @see MockitoTestExecutionListener - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link org.springframework.test.context.bean.override.mockito.MockitoResetTestExecutionListener} */ @SuppressWarnings("removal") diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpringBootMockResolver.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpringBootMockResolver.java index c78ce88b213e..43d0cca6b0a6 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpringBootMockResolver.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpringBootMockResolver.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ * * @author Andy Wilkinson * @since 2.4.0 - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of Spring Framework's + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of Spring Framework's * {@link MockitoBean} and {@link MockitoSpyBean} */ @Deprecated(since = "3.4.0", forRemoval = true) @@ -44,7 +44,7 @@ public Object resolve(Object instance) { @SuppressWarnings("unchecked") private static T getUltimateTargetObject(Object candidate) { - Assert.notNull(candidate, "Candidate must not be null"); + Assert.notNull(candidate, "'candidate' must not be null"); try { if (AopUtils.isAopProxy(candidate) && candidate instanceof Advised advised) { TargetSource targetSource = advised.getTargetSource(); diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyBean.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyBean.java index 2ce011caef79..a5aaadc9387b 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyBean.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyBean.java @@ -89,7 +89,7 @@ * @author Phillip Webb * @since 1.4.0 * @see MockitoPostProcessor - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link org.springframework.test.context.bean.override.mockito.MockitoSpyBean} */ @SuppressWarnings("removal") diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyBeans.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyBeans.java index 4a0b6d6a1e17..5e8cbdb37b5d 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyBeans.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyBeans.java @@ -33,7 +33,7 @@ * * @author Phillip Webb * @since 1.4.0 - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link org.springframework.test.context.bean.override.mockito.MockitoSpyBean} */ @SuppressWarnings("removal") diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyDefinition.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyDefinition.java index 5a9bb64f5b3e..1c06075b31b7 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyDefinition.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/SpyDefinition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ * A complete definition that can be used to create a Mockito spy. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) @@ -50,7 +50,7 @@ class SpyDefinition extends Definition { SpyDefinition(String name, ResolvableType typeToSpy, MockReset reset, boolean proxyTargetAware, QualifierDefinition qualifier) { super(name, reset, proxyTargetAware, qualifier); - Assert.notNull(typeToSpy, "TypeToSpy must not be null"); + Assert.notNull(typeToSpy, "'typeToSpy' must not be null"); this.typeToSpy = typeToSpy; } @@ -94,7 +94,7 @@ T createSpy(Object instance) { @SuppressWarnings("unchecked") T createSpy(String name, Object instance) { - Assert.notNull(instance, "Instance must not be null"); + Assert.notNull(instance, "'instance' must not be null"); Assert.isInstanceOf(this.typeToSpy.resolve(), instance); if (Mockito.mockingDetails(instance).isSpy()) { return (T) instance; diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/package-info.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/package-info.java index 83f0038bac43..21f0e1581d73 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/package-info.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/mock/mockito/package-info.java @@ -17,7 +17,7 @@ /** * Mockito integration for Spring Boot tests. *

- * Deprecated since 3.4.0 for removal in 3.6.0 in favor of Spring Framework's + * Deprecated since 3.4.0 for removal in 4.0.0 in favor of Spring Framework's * {@link org.springframework.test.context.bean.override.mockito.MockitoBean} and * {@link org.springframework.test.context.bean.override.mockito.MockitoSpyBean} */ diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/util/TestPropertyValues.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/util/TestPropertyValues.java index 3a516ebe40b3..6d1e50e14e36 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/util/TestPropertyValues.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/util/TestPropertyValues.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -155,9 +155,9 @@ public void applyTo(ConfigurableEnvironment environment, Type type) { * @param name the name for the property source */ public void applyTo(ConfigurableEnvironment environment, Type type, String name) { - Assert.notNull(environment, "Environment must not be null"); - Assert.notNull(type, "Property source type must not be null"); - Assert.notNull(name, "Property source name must not be null"); + Assert.notNull(environment, "'environment' must not be null"); + Assert.notNull(type, "'type' must not be null"); + Assert.notNull(name, "'name' must not be null"); MutablePropertySources sources = environment.getPropertySources(); addToSources(sources, type, name); ConfigurationPropertySources.attach(environment); @@ -324,7 +324,7 @@ public static final class Pair { private final String value; private Pair(String name, String value) { - Assert.hasLength(name, "Name must not be empty"); + Assert.hasLength(name, "'name' must not be empty"); this.name = name; this.value = value; } @@ -401,7 +401,7 @@ public void close() { } private String setOrClear(String name, String value) { - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); if (!StringUtils.hasLength(value)) { return (String) System.getProperties().remove(name); } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/LocalHostUriTemplateHandler.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/LocalHostUriTemplateHandler.java index 035779d2ec8c..41ab878c721f 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/LocalHostUriTemplateHandler.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/LocalHostUriTemplateHandler.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -72,8 +72,8 @@ public LocalHostUriTemplateHandler(Environment environment, String scheme) { */ public LocalHostUriTemplateHandler(Environment environment, String scheme, UriTemplateHandler handler) { super(handler); - Assert.notNull(environment, "Environment must not be null"); - Assert.notNull(scheme, "Scheme must not be null"); + Assert.notNull(environment, "'environment' must not be null"); + Assert.notNull(scheme, "'scheme' must not be null"); this.environment = environment; this.scheme = scheme; } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/MockServerRestClientCustomizer.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/MockServerRestClientCustomizer.java index 7c98aa76dc9d..fef4c1af0697 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/MockServerRestClientCustomizer.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/MockServerRestClientCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -80,7 +80,7 @@ public MockServerRestClientCustomizer() { */ public MockServerRestClientCustomizer(Class expectationManager) { this(() -> BeanUtils.instantiateClass(expectationManager)); - Assert.notNull(expectationManager, "ExpectationManager must not be null"); + Assert.notNull(expectationManager, "'expectationManager' must not be null"); } /** @@ -90,7 +90,7 @@ public MockServerRestClientCustomizer(Class * @since 3.0.0 */ public MockServerRestClientCustomizer(Supplier expectationManagerSupplier) { - Assert.notNull(expectationManagerSupplier, "ExpectationManagerSupplier must not be null"); + Assert.notNull(expectationManagerSupplier, "'expectationManagerSupplier' must not be null"); this.expectationManagerSupplier = expectationManagerSupplier; } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/MockServerRestTemplateCustomizer.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/MockServerRestTemplateCustomizer.java index 3cd206bc1a79..985fce7ce1be 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/MockServerRestTemplateCustomizer.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/MockServerRestTemplateCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -82,7 +82,7 @@ public MockServerRestTemplateCustomizer() { */ public MockServerRestTemplateCustomizer(Class expectationManager) { this(() -> BeanUtils.instantiateClass(expectationManager)); - Assert.notNull(expectationManager, "ExpectationManager must not be null"); + Assert.notNull(expectationManager, "'expectationManager' must not be null"); } /** @@ -92,7 +92,7 @@ public MockServerRestTemplateCustomizer(Class expectationManagerSupplier) { - Assert.notNull(expectationManagerSupplier, "ExpectationManagerSupplier must not be null"); + Assert.notNull(expectationManagerSupplier, "'expectationManagerSupplier' must not be null"); this.expectationManagerSupplier = expectationManagerSupplier; } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/RootUriRequestExpectationManager.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/RootUriRequestExpectationManager.java index 6905c74e835f..5912cb6f223c 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/RootUriRequestExpectationManager.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/RootUriRequestExpectationManager.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -61,8 +61,8 @@ public class RootUriRequestExpectationManager implements RequestExpectationManag private final RequestExpectationManager expectationManager; public RootUriRequestExpectationManager(String rootUri, RequestExpectationManager expectationManager) { - Assert.notNull(rootUri, "RootUri must not be null"); - Assert.notNull(expectationManager, "ExpectationManager must not be null"); + Assert.notNull(rootUri, "'rootUri' must not be null"); + Assert.notNull(expectationManager, "'expectationManager' must not be null"); this.rootUri = rootUri; this.expectationManager = expectationManager; } @@ -156,7 +156,7 @@ public static MockRestServiceServer bindTo(RestTemplate restTemplate, */ public static RequestExpectationManager forRestTemplate(RestTemplate restTemplate, RequestExpectationManager expectationManager) { - Assert.notNull(restTemplate, "RestTemplate must not be null"); + Assert.notNull(restTemplate, "'restTemplate' must not be null"); UriTemplateHandler templateHandler = restTemplate.getUriTemplateHandler(); if (templateHandler instanceof RootUriTemplateHandler rootHandler) { return new RootUriRequestExpectationManager(rootHandler.getRootUri(), expectationManager); diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/TestRestTemplate.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/TestRestTemplate.java index c715b79fcd94..a966e509fe6c 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/TestRestTemplate.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/client/TestRestTemplate.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -154,7 +154,7 @@ private TestRestTemplate(RestTemplateBuilder builder, UriTemplateHandler uriTemp private static RestTemplateBuilder createInitialBuilder(RestTemplateBuilder builder, String username, String password, HttpClientOption... httpClientOptions) { - Assert.notNull(builder, "Builder must not be null"); + Assert.notNull(builder, "'builder' must not be null"); if (httpClientOptions != null) { ClientHttpRequestFactory requestFactory = builder.buildRequestFactory(); if (requestFactory instanceof HttpComponentsClientHttpRequestFactory) { @@ -1060,7 +1060,7 @@ protected static class CustomHttpComponentsClientHttpRequestFactory extends Http * Create a new {@link CustomHttpComponentsClientHttpRequestFactory} instance. * @param httpClientOptions the {@link HttpClient} options * @param settings the settings to apply - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link #CustomHttpComponentsClientHttpRequestFactory(HttpClientOption[], ClientHttpRequestFactorySettings)} */ @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/htmlunit/LocalHostWebClient.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/htmlunit/LocalHostWebClient.java index 47c46b477f45..dbc2cd2d35f7 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/htmlunit/LocalHostWebClient.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/htmlunit/LocalHostWebClient.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ public class LocalHostWebClient extends WebClient { private final Environment environment; public LocalHostWebClient(Environment environment) { - Assert.notNull(environment, "Environment must not be null"); + Assert.notNull(environment, "'environment' must not be null"); this.environment = environment; } diff --git a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/htmlunit/webdriver/LocalHostWebConnectionHtmlUnitDriver.java b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/htmlunit/webdriver/LocalHostWebConnectionHtmlUnitDriver.java index 60d0f7776d51..a2d7d6555c28 100644 --- a/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/htmlunit/webdriver/LocalHostWebConnectionHtmlUnitDriver.java +++ b/spring-boot-project/spring-boot-test/src/main/java/org/springframework/boot/test/web/htmlunit/webdriver/LocalHostWebConnectionHtmlUnitDriver.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,25 +35,25 @@ public class LocalHostWebConnectionHtmlUnitDriver extends WebConnectionHtmlUnitD private final Environment environment; public LocalHostWebConnectionHtmlUnitDriver(Environment environment) { - Assert.notNull(environment, "Environment must not be null"); + Assert.notNull(environment, "'environment' must not be null"); this.environment = environment; } public LocalHostWebConnectionHtmlUnitDriver(Environment environment, boolean enableJavascript) { super(enableJavascript); - Assert.notNull(environment, "Environment must not be null"); + Assert.notNull(environment, "'environment' must not be null"); this.environment = environment; } public LocalHostWebConnectionHtmlUnitDriver(Environment environment, BrowserVersion browserVersion) { super(browserVersion); - Assert.notNull(environment, "Environment must not be null"); + Assert.notNull(environment, "'environment' must not be null"); this.environment = environment; } public LocalHostWebConnectionHtmlUnitDriver(Environment environment, Capabilities capabilities) { super(capabilities); - Assert.notNull(environment, "Environment must not be null"); + Assert.notNull(environment, "'environment' must not be null"); this.environment = environment; } diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/AnnotatedClassFinderTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/AnnotatedClassFinderTests.java index e0da18ec1262..0d760525d9d7 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/AnnotatedClassFinderTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/AnnotatedClassFinderTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,13 +37,13 @@ class AnnotatedClassFinderTests { @Test void findFromClassWhenSourceIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> this.finder.findFromClass((Class) null)) - .withMessageContaining("Source must not be null"); + .withMessageContaining("'source' must not be null"); } @Test void findFromPackageWhenSourceIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> this.finder.findFromPackage((String) null)) - .withMessageContaining("Source must not be null"); + .withMessageContaining("'source' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/SpringBootContextLoaderTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/SpringBootContextLoaderTests.java index f58b1cf832ff..ddd5e4289362 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/SpringBootContextLoaderTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/SpringBootContextLoaderTests.java @@ -65,6 +65,7 @@ * @author Stephane Nicoll * @author Scott Frederick * @author Madhura Bhave + * @author Sijun Yang */ class SpringBootContextLoaderTests { @@ -132,11 +133,6 @@ void multipleActiveProfiles() { assertThat(getActiveProfiles(MultipleActiveProfiles.class)).containsExactly("profile1", "profile2"); } - @Test - void activeProfileWithComma() { - assertThat(getActiveProfiles(ActiveProfileWithComma.class)).containsExactly("profile1,2"); - } - @Test // gh-28776 void testPropertyValuesShouldTakePrecedenceWhenInlinedPropertiesPresent() { TestContext context = new ExposedTestContextManager(SimpleConfig.class).getExposedTestContext(); @@ -342,14 +338,8 @@ static class MultipleActiveProfiles { } - @SpringBootTest(classes = Config.class) - @ActiveProfiles({ "profile1,2" }) - static class ActiveProfileWithComma { - - } - @SpringBootTest(properties = { "key=myValue" }, classes = Config.class) - @ActiveProfiles({ "profile1,2" }) + @ActiveProfiles({ "profile1" }) static class ActiveProfileWithInlinedProperties { } diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertProviderTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertProviderTests.java index 8547e0f2ca34..d4455c55a44c 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertProviderTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertProviderTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -64,14 +64,14 @@ void setup() { void getWhenTypeIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy( () -> ApplicationContextAssertProvider.get(null, ApplicationContext.class, this.mockContextSupplier)) - .withMessageContaining("Type must not be null"); + .withMessageContaining("'type' must not be null"); } @Test void getWhenTypeIsClassShouldThrowException() { assertThatIllegalArgumentException().isThrownBy( () -> ApplicationContextAssertProvider.get(null, ApplicationContext.class, this.mockContextSupplier)) - .withMessageContaining("Type must not be null"); + .withMessageContaining("'type' must not be null"); } @Test @@ -79,7 +79,7 @@ void getWhenContextTypeIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> ApplicationContextAssertProvider.get(TestAssertProviderApplicationContextClass.class, ApplicationContext.class, this.mockContextSupplier)) - .withMessageContaining("Type must be an interface"); + .withMessageContaining("'type' must be an interface"); } @Test @@ -87,7 +87,7 @@ void getWhenContextTypeIsClassShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> ApplicationContextAssertProvider.get(TestAssertProviderApplicationContext.class, null, this.mockContextSupplier)) - .withMessageContaining("ContextType must not be null"); + .withMessageContaining("'contextType' must not be null"); } @Test @@ -95,7 +95,7 @@ void getWhenSupplierIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> ApplicationContextAssertProvider.get(TestAssertProviderApplicationContext.class, StaticApplicationContext.class, this.mockContextSupplier)) - .withMessageContaining("ContextType must be an interface"); + .withMessageContaining("'contextType' must be an interface"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertTests.java index 1990c5c63389..68beabe910fe 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/assertj/ApplicationContextAssertTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -62,7 +62,7 @@ void cleanup() { @Test void createWhenApplicationContextIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new ApplicationContextAssert<>(null, null)) - .withMessageContaining("ApplicationContext must not be null"); + .withMessageContaining("'applicationContext' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/runner/ContextConsumerTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/runner/ContextConsumerTests.java index 370e9d72f85a..36153bc05a72 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/runner/ContextConsumerTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/context/runner/ContextConsumerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -70,7 +70,7 @@ void andThenWithNull() { ContextConsumer consumer = (context) -> { }; assertThatIllegalArgumentException().isThrownBy(() -> consumer.andThen(null)) - .withMessage("After must not be null"); + .withMessage("'after' must not be null"); } } diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/AbstractJsonMarshalTesterTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/AbstractJsonMarshalTesterTests.java index f680b5f78a33..ba54bc5e3192 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/AbstractJsonMarshalTesterTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/AbstractJsonMarshalTesterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -93,13 +93,13 @@ void writeMapShouldReturnJsonContent() throws Exception { void createWhenResourceLoadClassIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> createTester(null, ResolvableType.forClass(ExampleObject.class))) - .withMessageContaining("ResourceLoadClass must not be null"); + .withMessageContaining("'resourceLoadClass' must not be null"); } @Test void createWhenTypeIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> createTester(getClass(), null)) - .withMessageContaining("Type must not be null"); + .withMessageContaining("'type' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/BasicJsonTesterTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/BasicJsonTesterTests.java index 50602dd5c7fa..205aebe9ee7a 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/BasicJsonTesterTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/BasicJsonTesterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,7 +45,7 @@ class BasicJsonTesterTests { @Test void createWhenResourceLoadClassIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new BasicJsonTester(null)) - .withMessageContaining("ResourceLoadClass must not be null"); + .withMessageContaining("'resourceLoadClass' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/GsonTesterTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/GsonTesterTests.java index 62fe8f3df959..ea1927753258 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/GsonTesterTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/GsonTesterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,14 +37,14 @@ class GsonTesterTests extends AbstractJsonMarshalTesterTests { @Test void initFieldsWhenTestIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> GsonTester.initFields(null, new GsonBuilder().create())) - .withMessageContaining("TestInstance must not be null"); + .withMessageContaining("'testInstance' must not be null"); } @Test void initFieldsWhenMarshallerIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> GsonTester.initFields(new InitFieldsTestClass(), (Gson) null)) - .withMessageContaining("Marshaller must not be null"); + .withMessageContaining("'marshaller' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JacksonTesterTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JacksonTesterTests.java index 366ba5ae7bde..7be1e0964ad9 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JacksonTesterTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JacksonTesterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,14 +36,14 @@ class JacksonTesterTests extends AbstractJsonMarshalTesterTests { @Test void initFieldsWhenTestIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> JacksonTester.initFields(null, new ObjectMapper())) - .withMessageContaining("TestInstance must not be null"); + .withMessageContaining("'testInstance' must not be null"); } @Test void initFieldsWhenMarshallerIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> JacksonTester.initFields(new InitFieldsTestClass(), (ObjectMapper) null)) - .withMessageContaining("Marshaller must not be null"); + .withMessageContaining("'marshaller' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JsonContentTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JsonContentTests.java index 1c3b27265e8d..a9e8b456aed7 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JsonContentTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JsonContentTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,7 @@ class JsonContentTests { void createWhenResourceLoadClassIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new JsonContent(null, TYPE, JSON, Configuration.defaultConfiguration())) - .withMessageContaining("ResourceLoadClass must not be null"); + .withMessageContaining("'resourceLoadClass' must not be null"); } @Test @@ -47,14 +47,14 @@ void createWhenJsonIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy( () -> new JsonContent(getClass(), TYPE, null, Configuration.defaultConfiguration())) - .withMessageContaining("JSON must not be null"); + .withMessageContaining("'json' must not be null"); } @Test void createWhenConfigurationIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new JsonContent(getClass(), TYPE, JSON, null)) - .withMessageContaining("Configuration must not be null"); + .withMessageContaining("'configuration' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JsonbTesterTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JsonbTesterTests.java index 524196cea30b..9e77a73a3fce 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JsonbTesterTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/JsonbTesterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,14 +37,14 @@ class JsonbTesterTests extends AbstractJsonMarshalTesterTests { @Test void initFieldsWhenTestIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> JsonbTester.initFields(null, JsonbBuilder.create())) - .withMessageContaining("TestInstance must not be null"); + .withMessageContaining("'testInstance' must not be null"); } @Test void initFieldsWhenMarshallerIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> JsonbTester.initFields(new InitFieldsTestClass(), (Jsonb) null)) - .withMessageContaining("Marshaller must not be null"); + .withMessageContaining("'marshaller' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/ObjectContentTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/ObjectContentTests.java index 8d83219da636..ce34a9b3fba6 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/ObjectContentTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/json/ObjectContentTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ class ObjectContentTests { @Test void createWhenObjectIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new ObjectContent(TYPE, null)) - .withMessageContaining("Object must not be null"); + .withMessageContaining("'object' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/AbstractMockBeanOnGenericExtensionTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/AbstractMockBeanOnGenericExtensionTests.java index 539d5eeb6f2a..87332b0146d7 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/AbstractMockBeanOnGenericExtensionTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/AbstractMockBeanOnGenericExtensionTests.java @@ -20,7 +20,7 @@ * Concrete implementation of {@link AbstractMockBeanOnGenericTests}. * * @author Madhura Bhave - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/AbstractMockBeanOnGenericTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/AbstractMockBeanOnGenericTests.java index 1d01a90ee065..712287181934 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/AbstractMockBeanOnGenericTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/AbstractMockBeanOnGenericTests.java @@ -31,7 +31,7 @@ * @param type of thing * @param type of something * @author Madhura Bhave - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/DefinitionsParserTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/DefinitionsParserTests.java index 0758d11d9cc9..2e56abc00f97 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/DefinitionsParserTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/DefinitionsParserTests.java @@ -36,7 +36,7 @@ * Tests for {@link DefinitionsParser}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanContextCachingTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanContextCachingTests.java index 6251f2ccb3a9..bde0d2fcd736 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanContextCachingTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanContextCachingTests.java @@ -42,7 +42,7 @@ * Tests for application context caching when using {@link MockBean @MockBean}. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanForBeanFactoryIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanForBeanFactoryIntegrationTests.java index bab9e5859184..ff9c64859660 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanForBeanFactoryIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanForBeanFactoryIntegrationTests.java @@ -34,7 +34,7 @@ * Test {@link MockBean @MockBean} for a factory bean. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationClassForExistingBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationClassForExistingBeanIntegrationTests.java index cd016b7757aa..b9d12d8bf1f9 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationClassForExistingBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationClassForExistingBeanIntegrationTests.java @@ -35,7 +35,7 @@ * existing beans. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationClassForNewBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationClassForNewBeanIntegrationTests.java index 8b537cf5d6a2..1b7f964bf4b5 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationClassForNewBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationClassForNewBeanIntegrationTests.java @@ -34,7 +34,7 @@ * instances. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @ExtendWith(SpringExtension.class) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationFieldForExistingBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationFieldForExistingBeanIntegrationTests.java index 2558a095e721..cbf05d98c9c1 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationFieldForExistingBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationFieldForExistingBeanIntegrationTests.java @@ -35,7 +35,7 @@ * used to replace existing beans. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @ExtendWith(SpringExtension.class) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationFieldForNewBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationFieldForNewBeanIntegrationTests.java index 6e95d7a081f9..553d8ebbe8a0 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationFieldForNewBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnConfigurationFieldForNewBeanIntegrationTests.java @@ -34,7 +34,7 @@ * used to inject new mock instances. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @ExtendWith(SpringExtension.class) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnContextHierarchyIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnContextHierarchyIntegrationTests.java index ba5fad0a6e8f..1827e0e25f2e 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnContextHierarchyIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnContextHierarchyIntegrationTests.java @@ -34,7 +34,7 @@ * {@link ContextHierarchy @ContextHierarchy}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnScopedProxyTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnScopedProxyTests.java index 0ba686e67211..fb294f8ca1ff 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnScopedProxyTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnScopedProxyTests.java @@ -38,7 +38,7 @@ * * @author Phillip Webb * @see gh-5724 - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestClassForExistingBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestClassForExistingBeanIntegrationTests.java index 6051d06a1910..0edce24e2c0c 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestClassForExistingBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestClassForExistingBeanIntegrationTests.java @@ -34,7 +34,7 @@ * Test {@link MockBean @MockBean} on a test class can be used to replace existing beans. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestClassForNewBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestClassForNewBeanIntegrationTests.java index 9d14d1b7411a..f24c850c5269 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestClassForNewBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestClassForNewBeanIntegrationTests.java @@ -34,7 +34,7 @@ * instances. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanCacheIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanCacheIntegrationTests.java index 6e8bbc80a199..78e12761a012 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanCacheIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanCacheIntegrationTests.java @@ -36,7 +36,7 @@ * * @author Phillip Webb * @see MockBeanOnTestFieldForExistingBeanIntegrationTests - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanConfig.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanConfig.java index 1210206ad095..cbe41c94f032 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanConfig.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanConfig.java @@ -27,7 +27,7 @@ * config to trigger caching. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanIntegrationTests.java index 1b9416c18fce..3e86bc87a1a0 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanIntegrationTests.java @@ -34,7 +34,7 @@ * * @author Phillip Webb * @see MockBeanOnTestFieldForExistingBeanCacheIntegrationTests - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanWithQualifierIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanWithQualifierIntegrationTests.java index d2cb657f2751..d105546e3e6a 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanWithQualifierIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForExistingBeanWithQualifierIntegrationTests.java @@ -39,7 +39,7 @@ * * @author Stephane Nicoll * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForNewBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForNewBeanIntegrationTests.java index b534ccc743af..a192da83c6c6 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForNewBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanOnTestFieldForNewBeanIntegrationTests.java @@ -34,7 +34,7 @@ * instances. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithAopProxyTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithAopProxyTests.java index a70788e23c6b..de40e7d1c3f8 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithAopProxyTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithAopProxyTests.java @@ -45,7 +45,7 @@ * * @author Phillip Webb * @see 5837 - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithAsyncInterfaceMethodIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithAsyncInterfaceMethodIntegrationTests.java index 4d062f934d9b..c12f8d1fb254 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithAsyncInterfaceMethodIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithAsyncInterfaceMethodIntegrationTests.java @@ -33,7 +33,7 @@ * Tests for a mock bean where the mocked interface has an async method. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithDirtiesContextClassModeBeforeMethodIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithDirtiesContextClassModeBeforeMethodIntegrationTests.java index a3a68c1d3090..6f3e47f7604a 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithDirtiesContextClassModeBeforeMethodIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithDirtiesContextClassModeBeforeMethodIntegrationTests.java @@ -36,7 +36,7 @@ * {@link DirtiesContext @DirtiesContext} and {@link ClassMode#BEFORE_EACH_TEST_METHOD}. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithGenericsOnTestFieldForNewBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithGenericsOnTestFieldForNewBeanIntegrationTests.java index 8553b5cddebb..f6e353c0e654 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithGenericsOnTestFieldForNewBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithGenericsOnTestFieldForNewBeanIntegrationTests.java @@ -34,7 +34,7 @@ * instances. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithInjectedFieldIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithInjectedFieldIntegrationTests.java index 7ff0d6e082a9..5c62dfe5a61b 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithInjectedFieldIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithInjectedFieldIntegrationTests.java @@ -31,7 +31,7 @@ * Tests for a mock bean where the class being mocked uses field injection. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithSpringMethodRuleRepeatJUnit4IntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithSpringMethodRuleRepeatJUnit4IntegrationTests.java index 43576ada919d..7fcff8fced2a 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithSpringMethodRuleRepeatJUnit4IntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockBeanWithSpringMethodRuleRepeatJUnit4IntegrationTests.java @@ -30,7 +30,7 @@ * * @author Andy Wilkinson * @see gh-27693 - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockDefinitionTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockDefinitionTests.java index f349e9e8a46e..c34800b1ce13 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockDefinitionTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockDefinitionTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,7 +33,7 @@ * Tests for {@link MockDefinition}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) @@ -45,7 +45,7 @@ class MockDefinitionTests { void classToMockMustNotBeNull() { assertThatIllegalArgumentException() .isThrownBy(() -> new MockDefinition(null, null, null, null, false, null, null)) - .withMessageContaining("TypeToMock must not be null"); + .withMessageContaining("'typeToMock' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockResetTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockResetTests.java index a2cddb61a788..bc90e1fc3fe7 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockResetTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockResetTests.java @@ -28,7 +28,7 @@ * Tests for {@link MockReset}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerFactoryTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerFactoryTests.java index 4081af4e36f2..260159ad6977 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerFactoryTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerFactoryTests.java @@ -26,7 +26,7 @@ * Tests for {@link MockitoContextCustomizerFactory}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerTests.java index 203b4288c311..09003c94b2f2 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoContextCustomizerTests.java @@ -33,7 +33,7 @@ * Tests for {@link MockitoContextCustomizer}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoPostProcessorTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoPostProcessorTests.java index e492abc0e90d..50e814d428e9 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoPostProcessorTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoPostProcessorTests.java @@ -50,7 +50,7 @@ * @author Andy Wilkinson * @author Andreas Neiser * @author Madhura Bhave - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListenerIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListenerIntegrationTests.java index 7804ffdecd7d..eeba66c3ab66 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListenerIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListenerIntegrationTests.java @@ -49,7 +49,7 @@ * Integration tests for {@link MockitoTestExecutionListener}. * * @author Moritz Halbritter - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListenerTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListenerTests.java index 09dec0c271c4..fe4da63cf0f3 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListenerTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/MockitoTestExecutionListenerTests.java @@ -41,7 +41,7 @@ * Tests for {@link MockitoTestExecutionListener}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/QualifierDefinitionTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/QualifierDefinitionTests.java index 4b7fdeaffaa6..9f7cb1afc8ed 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/QualifierDefinitionTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/QualifierDefinitionTests.java @@ -40,7 +40,7 @@ * Tests for {@link QualifierDefinition}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/ResetMocksTestExecutionListenerTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/ResetMocksTestExecutionListenerTests.java index 2aa18248079c..7801392aa21b 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/ResetMocksTestExecutionListenerTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/ResetMocksTestExecutionListenerTests.java @@ -39,7 +39,7 @@ * * @author Phillip Webb * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpringBootMockResolverTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpringBootMockResolverTests.java index 138bf78ae132..f28aa3ebf326 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpringBootMockResolverTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpringBootMockResolverTests.java @@ -29,7 +29,7 @@ * Tests for {@link SpringBootMockResolver}. * * @author Moritz Halbritter - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationClassForExistingBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationClassForExistingBeanIntegrationTests.java index 2b424acac20a..aa0e93ed840b 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationClassForExistingBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationClassForExistingBeanIntegrationTests.java @@ -34,7 +34,7 @@ * beans. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationClassForNewBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationClassForNewBeanIntegrationTests.java index 10049592428a..5a328b282fdd 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationClassForNewBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationClassForNewBeanIntegrationTests.java @@ -34,7 +34,7 @@ * instances. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationFieldForExistingBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationFieldForExistingBeanIntegrationTests.java index ed0c862b6735..293d418e0afb 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationFieldForExistingBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationFieldForExistingBeanIntegrationTests.java @@ -35,7 +35,7 @@ * to replace existing beans. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationFieldForNewBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationFieldForNewBeanIntegrationTests.java index 9a85a37b3c2d..23aca392ebc3 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationFieldForNewBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnConfigurationFieldForNewBeanIntegrationTests.java @@ -34,7 +34,7 @@ * to inject new spy instances. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnContextHierarchyIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnContextHierarchyIntegrationTests.java index b9763345da07..38da81da1b1b 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnContextHierarchyIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnContextHierarchyIntegrationTests.java @@ -34,7 +34,7 @@ * {@link ContextHierarchy @ContextHierarchy}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestClassForExistingBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestClassForExistingBeanIntegrationTests.java index c3db640893a5..ee008a55090c 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestClassForExistingBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestClassForExistingBeanIntegrationTests.java @@ -33,7 +33,7 @@ * Test {@link SpyBean @SpyBean} on a test class can be used to replace existing beans. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestClassForNewBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestClassForNewBeanIntegrationTests.java index 90bc40ea7040..ba1c9ce0131e 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestClassForNewBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestClassForNewBeanIntegrationTests.java @@ -33,7 +33,7 @@ * Test {@link SpyBean @SpyBean} on a test class can be used to inject new spy instances. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanCacheIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanCacheIntegrationTests.java index 77139b368083..44eb0c543f04 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanCacheIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanCacheIntegrationTests.java @@ -36,7 +36,7 @@ * * @author Phillip Webb * @see SpyBeanOnTestFieldForExistingBeanIntegrationTests - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanConfig.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanConfig.java index 384fb4fb26a9..469d0c2c4f63 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanConfig.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanConfig.java @@ -27,7 +27,7 @@ * config to trigger caching. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanIntegrationTests.java index a1a0050986cb..0645169c62fb 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanIntegrationTests.java @@ -34,7 +34,7 @@ * * @author Phillip Webb * @see SpyBeanOnTestFieldForExistingBeanCacheIntegrationTests - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanWithQualifierIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanWithQualifierIntegrationTests.java index ce3de45b7056..1f2b27c56137 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanWithQualifierIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingBeanWithQualifierIntegrationTests.java @@ -38,7 +38,7 @@ * bean while preserving qualifiers. * * @author Andreas Neiser - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingCircularBeansIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingCircularBeansIntegrationTests.java index 2a933eada63d..f6437a67e34d 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingCircularBeansIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingCircularBeansIntegrationTests.java @@ -31,7 +31,7 @@ * beans with circular dependencies. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingGenericBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingGenericBeanIntegrationTests.java index 48a4b244abf3..b8dfc4ee2263 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingGenericBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingGenericBeanIntegrationTests.java @@ -38,7 +38,7 @@ * * @author Phillip Webb * @see SpyBeanOnTestFieldForExistingBeanCacheIntegrationTests - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingGenericBeanProducedByFactoryBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingGenericBeanProducedByFactoryBeanIntegrationTests.java index 338465332a03..5711ba771c70 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingGenericBeanProducedByFactoryBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForExistingGenericBeanProducedByFactoryBeanIntegrationTests.java @@ -39,7 +39,7 @@ * bean with generics that's produced by a factory bean. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForMultipleExistingBeansWithOnePrimaryIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForMultipleExistingBeansWithOnePrimaryIntegrationTests.java index 538925c6f23c..073c2c78c1b6 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForMultipleExistingBeansWithOnePrimaryIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForMultipleExistingBeansWithOnePrimaryIntegrationTests.java @@ -37,7 +37,7 @@ * instance when there are multiple candidates and one is primary. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForNewBeanIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForNewBeanIntegrationTests.java index f1c2f51df381..3a2e5c29da14 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForNewBeanIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanOnTestFieldForNewBeanIntegrationTests.java @@ -34,7 +34,7 @@ * instances. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithAopProxyAndNotProxyTargetAwareTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithAopProxyAndNotProxyTargetAwareTests.java index aaaee7f7ac4e..4b9cf9dd69bd 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithAopProxyAndNotProxyTargetAwareTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithAopProxyAndNotProxyTargetAwareTests.java @@ -43,7 +43,7 @@ * * @author Phillip Webb * @see 5837 - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithAopProxyTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithAopProxyTests.java index edeb01bdf4fc..402afcb65e29 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithAopProxyTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithAopProxyTests.java @@ -43,7 +43,7 @@ * * @author Phillip Webb * @see 5837 - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithDirtiesContextClassModeBeforeMethodIntegrationTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithDirtiesContextClassModeBeforeMethodIntegrationTests.java index ca495fcbc213..c270a612d068 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithDirtiesContextClassModeBeforeMethodIntegrationTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithDirtiesContextClassModeBeforeMethodIntegrationTests.java @@ -35,7 +35,7 @@ * {@link DirtiesContext @DirtiesContext} and {@link ClassMode#BEFORE_EACH_TEST_METHOD}. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithJdkProxyTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithJdkProxyTests.java index 62a3c2874f7e..ae8e2debaa54 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithJdkProxyTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithJdkProxyTests.java @@ -34,7 +34,7 @@ * Tests for {@link SpyBean @SpyBean} with a JDK proxy. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithNameOnTestFieldForMultipleExistingBeansTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithNameOnTestFieldForMultipleExistingBeansTests.java index 518b2ac83692..c8251df5d0e9 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithNameOnTestFieldForMultipleExistingBeansTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyBeanWithNameOnTestFieldForMultipleExistingBeansTests.java @@ -34,7 +34,7 @@ * * @author Phillip Webb * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyDefinitionTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyDefinitionTests.java index 8f5cea4d6f08..f7416309e3f2 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyDefinitionTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/SpyDefinitionTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ * Tests for {@link SpyDefinition}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) @@ -45,7 +45,7 @@ class SpyDefinitionTests { @Test void classToSpyMustNotBeNull() { assertThatIllegalArgumentException().isThrownBy(() -> new SpyDefinition(null, null, null, true, null)) - .withMessageContaining("TypeToSpy must not be null"); + .withMessageContaining("'typeToSpy' must not be null"); } @Test @@ -84,7 +84,7 @@ void createSpy() { void createSpyWhenNullInstanceShouldThrowException() { SpyDefinition definition = new SpyDefinition("name", REAL_SERVICE_TYPE, MockReset.BEFORE, true, null); assertThatIllegalArgumentException().isThrownBy(() -> definition.createSpy(null)) - .withMessageContaining("Instance must not be null"); + .withMessageContaining("'instance' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/CustomQualifier.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/CustomQualifier.java index 9308e6940e30..7c9a4f0fa4bc 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/CustomQualifier.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/CustomQualifier.java @@ -25,7 +25,7 @@ * Custom qualifier for testing. * * @author Stephane Nicoll - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @Deprecated(since = "3.4.0", forRemoval = true) @Qualifier diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/CustomQualifierExampleService.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/CustomQualifierExampleService.java index d2505f0f1983..20eec60f14ae 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/CustomQualifierExampleService.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/CustomQualifierExampleService.java @@ -20,7 +20,7 @@ * An {@link ExampleService} that uses a custom qualifier. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleExtraInterface.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleExtraInterface.java index e2577ae2850e..3b5371c6fff0 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleExtraInterface.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleExtraInterface.java @@ -20,7 +20,7 @@ * Example extra interface for mocking tests. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @Deprecated(since = "3.4.0", forRemoval = true) public interface ExampleExtraInterface { diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericService.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericService.java index 5f9efa1b2d07..b8c43ee16472 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericService.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericService.java @@ -21,7 +21,7 @@ * * @param the generic type * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @Deprecated(since = "3.4.0", forRemoval = true) public interface ExampleGenericService { diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericServiceCaller.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericServiceCaller.java index 9a1c396c13e0..f199167d08de 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericServiceCaller.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericServiceCaller.java @@ -20,7 +20,7 @@ * Example bean for mocking tests that calls {@link ExampleGenericService}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericStringServiceCaller.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericStringServiceCaller.java index 859d3e1a2e1e..43e5eb7140ea 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericStringServiceCaller.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleGenericStringServiceCaller.java @@ -20,7 +20,7 @@ * Example bean for mocking tests that calls {@link ExampleGenericService}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleService.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleService.java index 15089baeda00..e23b08fcb9c4 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleService.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleService.java @@ -20,7 +20,7 @@ * Example service interface for mocking tests. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @Deprecated(since = "3.4.0", forRemoval = true) public interface ExampleService { diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleServiceCaller.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleServiceCaller.java index 36b3b06b8a8f..1b09ab593fe5 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleServiceCaller.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/ExampleServiceCaller.java @@ -20,7 +20,7 @@ * Example bean for mocking tests that calls {@link ExampleService}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/FailingExampleService.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/FailingExampleService.java index 9a96c3565e2f..5669aea08eb9 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/FailingExampleService.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/FailingExampleService.java @@ -22,7 +22,7 @@ * An {@link ExampleService} that always throws an exception. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/RealExampleService.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/RealExampleService.java index 56d3674c0350..7cdddd94327b 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/RealExampleService.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/RealExampleService.java @@ -20,7 +20,7 @@ * Example service implementation for spy tests. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleIntegerGenericService.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleIntegerGenericService.java index 0aa5b9e359b4..985bc624ea5b 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleIntegerGenericService.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleIntegerGenericService.java @@ -20,7 +20,7 @@ * Example generic service implementation for spy tests. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleService.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleService.java index 4ddb53e601b8..c4cbd52e7a13 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleService.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleService.java @@ -20,7 +20,7 @@ * Example service implementation for spy tests. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleStringGenericService.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleStringGenericService.java index a89b6cb1c671..f216be45c838 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleStringGenericService.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/mock/mockito/example/SimpleExampleStringGenericService.java @@ -20,7 +20,7 @@ * Example generic service implementation for spy tests. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/LocalHostUriTemplateHandlerTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/LocalHostUriTemplateHandlerTests.java index 48b63138376a..57370a9df58b 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/LocalHostUriTemplateHandlerTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/LocalHostUriTemplateHandlerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,21 +43,21 @@ class LocalHostUriTemplateHandlerTests { @Test void createWhenEnvironmentIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new LocalHostUriTemplateHandler(null)) - .withMessageContaining("Environment must not be null"); + .withMessageContaining("'environment' must not be null"); } @Test void createWhenSchemeIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new LocalHostUriTemplateHandler(new MockEnvironment(), null)) - .withMessageContaining("Scheme must not be null"); + .withMessageContaining("'scheme' must not be null"); } @Test void createWhenHandlerIsNullShouldThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new LocalHostUriTemplateHandler(new MockEnvironment(), "http", null)) - .withMessageContaining("Handler must not be null"); + .withMessageContaining("'handler' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/MockServerRestClientCustomizerTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/MockServerRestClientCustomizerTests.java index 0dffb76514a8..82469c395fea 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/MockServerRestClientCustomizerTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/MockServerRestClientCustomizerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -59,7 +59,7 @@ void createShouldUseSimpleRequestExpectationManager() { void createWhenExpectationManagerClassIsNullShouldThrowException() { Class expectationManager = null; assertThatIllegalArgumentException().isThrownBy(() -> new MockServerRestClientCustomizer(expectationManager)) - .withMessageContaining("ExpectationManager must not be null"); + .withMessageContaining("'expectationManager' must not be null"); } @Test @@ -67,7 +67,7 @@ void createWhenExpectationManagerSupplierIsNullShouldThrowException() { Supplier expectationManagerSupplier = null; assertThatIllegalArgumentException() .isThrownBy(() -> new MockServerRestClientCustomizer(expectationManagerSupplier)) - .withMessageContaining("ExpectationManagerSupplier must not be null"); + .withMessageContaining("'expectationManagerSupplier' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/MockServerRestTemplateCustomizerTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/MockServerRestTemplateCustomizerTests.java index 8f08c63abc94..c6787d202c72 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/MockServerRestTemplateCustomizerTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/MockServerRestTemplateCustomizerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -62,7 +62,7 @@ void createShouldUseSimpleRequestExpectationManager() { void createWhenExpectationManagerClassIsNullShouldThrowException() { Class expectationManager = null; assertThatIllegalArgumentException().isThrownBy(() -> new MockServerRestTemplateCustomizer(expectationManager)) - .withMessageContaining("ExpectationManager must not be null"); + .withMessageContaining("'expectationManager' must not be null"); } @Test @@ -70,7 +70,7 @@ void createWhenExpectationManagerSupplierIsNullShouldThrowException() { Supplier expectationManagerSupplier = null; assertThatIllegalArgumentException() .isThrownBy(() -> new MockServerRestTemplateCustomizer(expectationManagerSupplier)) - .withMessageContaining("ExpectationManagerSupplier must not be null"); + .withMessageContaining("'expectationManagerSupplier' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/RootUriRequestExpectationManagerTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/RootUriRequestExpectationManagerTests.java index a818798407fe..69b704761127 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/RootUriRequestExpectationManagerTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/client/RootUriRequestExpectationManagerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -67,13 +67,13 @@ void setup() { @Test void createWhenRootUriIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new RootUriRequestExpectationManager(null, this.delegate)) - .withMessageContaining("RootUri must not be null"); + .withMessageContaining("'rootUri' must not be null"); } @Test void createWhenExpectationManagerIsNullShouldThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new RootUriRequestExpectationManager(this.uri, null)) - .withMessageContaining("ExpectationManager must not be null"); + .withMessageContaining("'expectationManager' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/htmlunit/LocalHostWebClientTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/htmlunit/LocalHostWebClientTests.java index c3299a417515..84e0ff4f2980 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/htmlunit/LocalHostWebClientTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/htmlunit/LocalHostWebClientTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,7 @@ class LocalHostWebClientTests { @Test void createWhenEnvironmentIsNullWillThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new LocalHostWebClient(null)) - .withMessageContaining("Environment must not be null"); + .withMessageContaining("'environment' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/htmlunit/webdriver/LocalHostWebConnectionHtmlUnitDriverTests.java b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/htmlunit/webdriver/LocalHostWebConnectionHtmlUnitDriverTests.java index 9b159132054f..21e787a83c03 100644 --- a/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/htmlunit/webdriver/LocalHostWebConnectionHtmlUnitDriverTests.java +++ b/spring-boot-project/spring-boot-test/src/test/java/org/springframework/boot/test/web/htmlunit/webdriver/LocalHostWebConnectionHtmlUnitDriverTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -64,20 +64,20 @@ class LocalHostWebConnectionHtmlUnitDriverTests { @Test void createWhenEnvironmentIsNullWillThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new LocalHostWebConnectionHtmlUnitDriver(null)) - .withMessageContaining("Environment must not be null"); + .withMessageContaining("'environment' must not be null"); } @Test void createWithJavascriptFlagWhenEnvironmentIsNullWillThrowException() { assertThatIllegalArgumentException().isThrownBy(() -> new LocalHostWebConnectionHtmlUnitDriver(null, true)) - .withMessageContaining("Environment must not be null"); + .withMessageContaining("'environment' must not be null"); } @Test void createWithBrowserVersionWhenEnvironmentIsNullWillThrowException() { assertThatIllegalArgumentException() .isThrownBy(() -> new LocalHostWebConnectionHtmlUnitDriver(null, BrowserVersion.CHROME)) - .withMessageContaining("Environment must not be null"); + .withMessageContaining("'environment' must not be null"); } @Test @@ -87,7 +87,7 @@ void createWithCapabilitiesWhenEnvironmentIsNullWillThrowException() { given(capabilities.getBrowserVersion()).willReturn("chrome"); assertThatIllegalArgumentException() .isThrownBy(() -> new LocalHostWebConnectionHtmlUnitDriver(null, capabilities)) - .withMessageContaining("Environment must not be null"); + .withMessageContaining("'environment' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-testcontainers/build.gradle b/spring-boot-project/spring-boot-testcontainers/build.gradle index cdbe496ae53b..73bbbcf974e6 100644 --- a/spring-boot-project/spring-boot-testcontainers/build.gradle +++ b/spring-boot-project/spring-boot-testcontainers/build.gradle @@ -70,6 +70,7 @@ dependencies { optional("org.testcontainers:grafana") optional("org.testcontainers:jdbc") optional("org.testcontainers:kafka") + optional("org.testcontainers:ldap") optional("org.testcontainers:mariadb") optional("org.testcontainers:mongodb") optional("org.testcontainers:mssqlserver") diff --git a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/cassandra/DeprecatedCassandraContainerConnectionDetailsFactoryTests.java b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/cassandra/DeprecatedCassandraContainerConnectionDetailsFactoryTests.java index 6d13f8edddcc..4d554b81c16d 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/cassandra/DeprecatedCassandraContainerConnectionDetailsFactoryTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/cassandra/DeprecatedCassandraContainerConnectionDetailsFactoryTests.java @@ -37,7 +37,7 @@ * Tests for {@link DeprecatedCassandraContainerConnectionDetailsFactory}. * * @author Andy Wilkinson - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SpringJUnitConfig @Testcontainers(disabledWithoutDocker = true) diff --git a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/elasticsearch/ElasticsearchContainerConnectionDetailsFactoryTests.java b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/elasticsearch/ElasticsearchContainerConnectionDetailsFactoryTests.java index be32783368e2..212c47757c70 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/elasticsearch/ElasticsearchContainerConnectionDetailsFactoryTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/elasticsearch/ElasticsearchContainerConnectionDetailsFactoryTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,6 +17,7 @@ package org.springframework.boot.testcontainers.service.connection.elasticsearch; import java.io.IOException; +import java.time.Duration; import co.elastic.clients.elasticsearch.ElasticsearchClient; import org.junit.jupiter.api.Test; @@ -30,7 +31,7 @@ import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchConnectionDetails; import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchRestClientAutoConfiguration; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; -import org.springframework.boot.testsupport.container.TestImage; +import org.springframework.boot.testsupport.container.ElasticsearchContainer8; import org.springframework.context.annotation.Configuration; import org.springframework.test.context.junit.jupiter.SpringJUnitConfig; @@ -47,7 +48,8 @@ class ElasticsearchContainerConnectionDetailsFactoryTests { @Container @ServiceConnection - static final ElasticsearchContainer elasticsearch = TestImage.container(ElasticsearchContainer.class); + static final ElasticsearchContainer elasticsearch = new ElasticsearchContainer8().withStartupAttempts(5) + .withStartupTimeout(Duration.ofMinutes(10)); @Autowired(required = false) private ElasticsearchConnectionDetails connectionDetails; diff --git a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/kafka/DeprecatedConfluentKafkaContainerConnectionDetailsFactoryIntegrationTests.java b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/kafka/DeprecatedConfluentKafkaContainerConnectionDetailsFactoryIntegrationTests.java index 3d9b0aedfe0f..b28c0c870c86 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/kafka/DeprecatedConfluentKafkaContainerConnectionDetailsFactoryIntegrationTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/kafka/DeprecatedConfluentKafkaContainerConnectionDetailsFactoryIntegrationTests.java @@ -46,7 +46,7 @@ * @author Moritz Halbritter * @author Andy Wilkinson * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SpringJUnitConfig @Testcontainers(disabledWithoutDocker = true) diff --git a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/ldap/LLdapContainerConnectionDetailsFactoryIntegrationTests.java b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/ldap/LLdapContainerConnectionDetailsFactoryIntegrationTests.java new file mode 100644 index 000000000000..3bb58c6e5738 --- /dev/null +++ b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/ldap/LLdapContainerConnectionDetailsFactoryIntegrationTests.java @@ -0,0 +1,68 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.testcontainers.service.connection.ldap; + +import java.util.List; + +import org.junit.jupiter.api.Test; +import org.testcontainers.junit.jupiter.Container; +import org.testcontainers.junit.jupiter.Testcontainers; +import org.testcontainers.ldap.LLdapContainer; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.autoconfigure.ImportAutoConfiguration; +import org.springframework.boot.autoconfigure.ldap.LdapAutoConfiguration; +import org.springframework.boot.testcontainers.service.connection.ServiceConnection; +import org.springframework.boot.testsupport.container.TestImage; +import org.springframework.context.annotation.Configuration; +import org.springframework.ldap.core.AttributesMapper; +import org.springframework.ldap.core.LdapTemplate; +import org.springframework.ldap.query.LdapQueryBuilder; +import org.springframework.test.context.junit.jupiter.SpringJUnitConfig; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link LLdapContainerConnectionDetailsFactory}. + * + * @author Eddú Meléndez + */ +@SpringJUnitConfig +@Testcontainers(disabledWithoutDocker = true) +class LLdapContainerConnectionDetailsFactoryIntegrationTests { + + @Container + @ServiceConnection + static final LLdapContainer lldap = TestImage.container(LLdapContainer.class); + + @Autowired + private LdapTemplate ldapTemplate; + + @Test + void connectionCanBeMadeToLdapContainer() { + List cn = this.ldapTemplate.search(LdapQueryBuilder.query().where("objectClass").is("inetOrgPerson"), + (AttributesMapper) (attributes) -> attributes.get("cn").get().toString()); + assertThat(cn).singleElement().isEqualTo("Administrator"); + } + + @Configuration(proxyBeanMethods = false) + @ImportAutoConfiguration({ LdapAutoConfiguration.class }) + static class TestConfiguration { + + } + +} diff --git a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/otlp/GrafanaOpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests.java b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/otlp/GrafanaOpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests.java index fed9b1ed0639..f510959e92fa 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/otlp/GrafanaOpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/otlp/GrafanaOpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -50,7 +50,7 @@ * @author Eddú Meléndez */ @SpringJUnitConfig -@TestPropertySource(properties = { "management.otlp.metrics.export.resource-attributes.service.name=test", +@TestPropertySource(properties = { "management.opentelemetry.resource-attributes.service.name=test", "management.otlp.metrics.export.step=1s" }) @Testcontainers(disabledWithoutDocker = true) class GrafanaOpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests { diff --git a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/otlp/OpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests.java b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/otlp/OpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests.java index 50b558711ebc..4bbaa91cf6a4 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/otlp/OpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/dockerTest/java/org/springframework/boot/testcontainers/service/connection/otlp/OpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -54,7 +54,7 @@ * @author Jonatan Ivanov */ @SpringJUnitConfig -@TestPropertySource(properties = { "management.otlp.metrics.export.resource-attributes.service.name=test", +@TestPropertySource(properties = { "management.opentelemetry.resource-attributes.service.name=test", "management.otlp.metrics.export.step=1s" }) @Testcontainers(disabledWithoutDocker = true) class OpenTelemetryMetricsContainerConnectionDetailsFactoryIntegrationTests { diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/lifecycle/BeforeTestcontainerUsedEvent.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/lifecycle/BeforeTestcontainerUsedEvent.java index eeab21286084..9b74b7f61f2b 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/lifecycle/BeforeTestcontainerUsedEvent.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/lifecycle/BeforeTestcontainerUsedEvent.java @@ -26,7 +26,7 @@ * * @author Andy Wilkinson * @since 3.2.6 - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of property registration using a + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of property registration using a * {@link DynamicPropertyRegistrar} bean that injects the {@link Container} from which the * properties will be sourced. */ diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/properties/TestcontainersPropertySource.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/properties/TestcontainersPropertySource.java index f9a5730242d6..b48e3fdcf779 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/properties/TestcontainersPropertySource.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/properties/TestcontainersPropertySource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -54,7 +54,7 @@ * * @author Phillip Webb * @since 3.1.0 - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of declaring one or more + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of declaring one or more * {@link DynamicPropertyRegistrar} beans. */ @SuppressWarnings("removal") @@ -77,7 +77,7 @@ private TestcontainersPropertySource(Map> valueSupplier DynamicPropertyRegistryInjection registryInjection) { super(NAME, Collections.unmodifiableMap(valueSuppliers)); this.registry = (name, valueSupplier) -> { - Assert.hasText(name, "'name' must not be null or blank"); + Assert.hasText(name, "'name' must not be empty"); DynamicPropertyRegistryInjectionException.throwIfNecessary(name, registryInjection); Assert.notNull(valueSupplier, "'valueSupplier' must not be null"); valueSuppliers.put(name, valueSupplier); diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionDetailsFactory.java index 14518b982d8b..6ad17a87c5b7 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,6 +16,7 @@ package org.springframework.boot.testcontainers.service.connection; +import java.lang.annotation.Annotation; import java.util.Arrays; import java.util.List; import java.util.stream.Stream; @@ -33,6 +34,7 @@ import org.springframework.boot.autoconfigure.service.connection.ConnectionDetailsFactory; import org.springframework.boot.origin.Origin; import org.springframework.boot.origin.OriginProvider; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.lifecycle.TestcontainersStartup; import org.springframework.context.ApplicationContext; import org.springframework.context.ApplicationContextAware; @@ -92,7 +94,7 @@ protected ContainerConnectionDetailsFactory(String connectionName, String... req * @since 3.4.0 */ protected ContainerConnectionDetailsFactory(List connectionNames, String... requiredClassNames) { - Assert.notEmpty(connectionNames, "ConnectionNames must contain at least one name"); + Assert.notEmpty(connectionNames, "'connectionNames' must not be empty"); this.connectionNames = connectionNames; this.requiredClassNames = requiredClassNames; } @@ -166,12 +168,14 @@ protected static class ContainerConnectionDetails> private volatile C container; + private volatile SslBundle sslBundle; + /** * Create a new {@link ContainerConnectionDetails} instance. * @param source the source {@link ContainerConnectionSource} */ protected ContainerConnectionDetails(ContainerConnectionSource source) { - Assert.notNull(source, "Source must not be null"); + Assert.notNull(source, "'source' must not be null"); this.source = source; } @@ -194,6 +198,33 @@ protected final C getContainer() { return this.container; } + /** + * Return the {@link SslBundle} to use with this connection or {@code null}. + * @return the ssl bundle or {@code null} + * @since 3.5.0 + */ + protected SslBundle getSslBundle() { + if (this.source.getSslBundleSource() == null) { + return null; + } + SslBundle sslBundle = this.sslBundle; + if (sslBundle == null) { + sslBundle = this.source.getSslBundleSource().getSslBundle(); + this.sslBundle = sslBundle; + } + return sslBundle; + } + + /** + * Whether the field or bean is annotated with the given annotation. + * @param annotationType the annotation to check + * @return whether the field or bean is annotated with the annotation + * @since 3.5.0 + */ + protected boolean hasAnnotation(Class annotationType) { + return this.source.hasAnnotation(annotationType); + } + @Override public Origin getOrigin() { return this.source.getOrigin(); diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionSource.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionSource.java index d589b4274dd6..598271fd62a6 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionSource.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionSource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,6 +16,7 @@ package org.springframework.boot.testcontainers.service.connection; +import java.lang.annotation.Annotation; import java.util.Set; import java.util.function.Supplier; @@ -27,6 +28,7 @@ import org.springframework.boot.origin.Origin; import org.springframework.boot.origin.OriginProvider; import org.springframework.core.annotation.MergedAnnotation; +import org.springframework.core.annotation.MergedAnnotations; import org.springframework.core.log.LogMessage; import org.springframework.util.StringUtils; @@ -60,8 +62,13 @@ public final class ContainerConnectionSource> implements private final Supplier containerSupplier; + private final SslBundleSource sslBundleSource; + + private final MergedAnnotations annotations; + ContainerConnectionSource(String beanNameSuffix, Origin origin, Class containerType, String containerImageName, - MergedAnnotation annotation, Supplier containerSupplier) { + MergedAnnotation annotation, Supplier containerSupplier, + SslBundleSource sslBundleSource, MergedAnnotations annotations) { this.beanNameSuffix = beanNameSuffix; this.origin = origin; this.containerType = containerType; @@ -69,10 +76,13 @@ public final class ContainerConnectionSource> implements this.connectionName = getOrDeduceConnectionName(annotation.getString("name"), containerImageName); this.connectionDetailsTypes = Set.of(annotation.getClassArray("type")); this.containerSupplier = containerSupplier; + this.sslBundleSource = sslBundleSource; + this.annotations = annotations; } ContainerConnectionSource(String beanNameSuffix, Origin origin, Class containerType, String containerImageName, - ServiceConnection annotation, Supplier containerSupplier) { + ServiceConnection annotation, Supplier containerSupplier, SslBundleSource sslBundleSource, + MergedAnnotations annotations) { this.beanNameSuffix = beanNameSuffix; this.origin = origin; this.containerType = containerType; @@ -80,6 +90,8 @@ public final class ContainerConnectionSource> implements this.connectionName = getOrDeduceConnectionName(annotation.name(), containerImageName); this.connectionDetailsTypes = Set.of(annotation.type()); this.containerSupplier = containerSupplier; + this.sslBundleSource = sslBundleSource; + this.annotations = annotations; } private static String getOrDeduceConnectionName(String connectionName, String containerImageName) { @@ -156,6 +168,17 @@ Set> getConnectionDetailsTypes() { return this.connectionDetailsTypes; } + SslBundleSource getSslBundleSource() { + return this.sslBundleSource; + } + + boolean hasAnnotation(Class annotationType) { + if (this.annotations == null) { + return false; + } + return this.annotations.isPresent(annotationType); + } + @Override public String toString() { return "@ServiceConnection source for %s".formatted(this.origin); diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/FieldOrigin.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/FieldOrigin.java index 3205aa4eb9f6..2bccf69beb14 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/FieldOrigin.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/FieldOrigin.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ class FieldOrigin implements Origin { private final Field field; FieldOrigin(Field field) { - Assert.notNull(field, "Field must not be null"); + Assert.notNull(field, "'field' must not be null"); this.field = field; } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/JksKeyStore.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/JksKeyStore.java new file mode 100644 index 000000000000..697379450977 --- /dev/null +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/JksKeyStore.java @@ -0,0 +1,71 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.testcontainers.service.connection; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.jks.JksSslStoreBundle; +import org.springframework.core.annotation.AliasFor; + +/** + * Configures the {@link JksSslStoreBundle} key store to use with an {@link SslBundle SSL} + * supported {@link ServiceConnection @ServiceConnection}. + * + * @author Phillip Webb + * @since 3.5.0 + */ +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.FIELD, ElementType.METHOD, ElementType.ANNOTATION_TYPE }) +public @interface JksKeyStore { + + /** + * Alias for {@link #location()}. + * @return the store location + */ + @AliasFor("location") + String value() default ""; + + /** + * The location of the resource containing the store content. + * @return the store location + */ + @AliasFor("value") + String location() default ""; + + /** + * The password used to access the store. + * @return the store password + */ + String password() default ""; + + /** + * The type of the store to create, e.g. JKS. + * @return store type + */ + String type() default ""; + + /** + * The provider for the store. + * @return the store provider + */ + String provider() default ""; + +} diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/JksTrustStore.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/JksTrustStore.java new file mode 100644 index 000000000000..2c8fd42bcd06 --- /dev/null +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/JksTrustStore.java @@ -0,0 +1,71 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.testcontainers.service.connection; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.jks.JksSslStoreBundle; +import org.springframework.core.annotation.AliasFor; + +/** + * Configures the {@link JksSslStoreBundle} trust store to use with an {@link SslBundle + * SSL} supported {@link ServiceConnection @ServiceConnection}. + * + * @author Phillip Webb + * @since 3.5.0 + */ +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.FIELD, ElementType.METHOD, ElementType.ANNOTATION_TYPE }) +public @interface JksTrustStore { + + /** + * Alias for {@link #location()}. + * @return the store location + */ + @AliasFor("location") + String value() default ""; + + /** + * The location of the resource containing the store content. + * @return the store location + */ + @AliasFor("value") + String location() default ""; + + /** + * The password used to access the store. + * @return the store password + */ + String password() default ""; + + /** + * The type of the store to create, e.g. JKS. + * @return store type + */ + String type() default ""; + + /** + * The provider for the store. + * @return the store provider + */ + String provider() default ""; + +} diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/PemKeyStore.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/PemKeyStore.java new file mode 100644 index 000000000000..a612dc25ecf4 --- /dev/null +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/PemKeyStore.java @@ -0,0 +1,71 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.testcontainers.service.connection; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.pem.PemSslStoreBundle; +import org.springframework.core.annotation.AliasFor; + +/** + * Configures the {@link PemSslStoreBundle} key store to use with an {@link SslBundle SSL} + * supported {@link ServiceConnection @ServiceConnection}. + * + * @author Phillip Webb + * @since 3.5.0 + */ +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.FIELD, ElementType.METHOD, ElementType.ANNOTATION_TYPE }) +public @interface PemKeyStore { + + /** + * Alias for {@link #certificate()}. + * @return the store certificate + */ + @AliasFor("certificate") + String value() default ""; + + /** + * The location or content of the certificate or certificate chain in PEM format. + * @return the store certificate location or content + */ + @AliasFor("value") + String certificate() default ""; + + /** + * The location or content of the private key in PEM format. + * @return the store private key location or content + */ + String privateKey() default ""; + + /** + * The password used to decrypt an encrypted private key. + * @return the store private key password + */ + String privateKeyPassword() default ""; + + /** + * The type of the store to create, e.g. JKS. + * @return the store type + */ + String type() default ""; + +} diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/PemTrustStore.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/PemTrustStore.java new file mode 100644 index 000000000000..187ad4944d1d --- /dev/null +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/PemTrustStore.java @@ -0,0 +1,71 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.testcontainers.service.connection; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.pem.PemSslStoreBundle; +import org.springframework.core.annotation.AliasFor; + +/** + * Configures the {@link PemSslStoreBundle} trust store to use with an {@link SslBundle + * SSL} supported {@link ServiceConnection @ServiceConnection}. + * + * @author Phillip Webb + * @since 3.5.0 + */ +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.FIELD, ElementType.METHOD, ElementType.ANNOTATION_TYPE }) +public @interface PemTrustStore { + + /** + * Alias for {@link #certificate()}. + * @return the store certificate + */ + @AliasFor("certificate") + String value() default ""; + + /** + * The location or content of the certificate or certificate chain in PEM format. + * @return the store certificate location or content + */ + @AliasFor("value") + String certificate() default ""; + + /** + * The location or content of the private key in PEM format. + * @return the store private key location or content + */ + String privateKey() default ""; + + /** + * The password used to decrypt an encrypted private key. + * @return the store private key password + */ + String privateKeyPassword() default ""; + + /** + * The type of the store to create, e.g. JKS. + * @return the store type + */ + String type() default ""; + +} diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnection.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnection.java index a6827dfd2cdb..248a67cae6da 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnection.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnection.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,8 +29,13 @@ import org.springframework.core.annotation.AliasFor; /** - * Annotation used to indicate that a field or method is a - * {@link ContainerConnectionSource} which provides a service that can be connected to. + * Indicates that a field or method is a {@link ContainerConnectionSource} which provides + * a service that can be connected to. + *

+ * If the underling connection supports SSL, the {@link PemKeyStore @PemKeyStore}, + * {@link PemTrustStore @PemTrustStore}, {@link JksKeyStore @JksKeyStore}, + * {@link JksTrustStore @JksTrustStore}, {@link Ssl @Ssl} annotations may be used to + * provide additional configuration. * * @author Moritz Halbritter * @author Andy Wilkinson diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionAutoConfigurationRegistrar.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionAutoConfigurationRegistrar.java index ae6c9b10ea43..144d2fa6776d 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionAutoConfigurationRegistrar.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionAutoConfigurationRegistrar.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,6 +31,7 @@ import org.springframework.boot.testcontainers.beans.TestcontainerBeanDefinition; import org.springframework.context.annotation.ImportBeanDefinitionRegistrar; import org.springframework.core.annotation.MergedAnnotation; +import org.springframework.core.annotation.MergedAnnotations; import org.springframework.core.type.AnnotationMetadata; /** @@ -59,24 +60,27 @@ private void registerBeanDefinitions(ConfigurableListableBeanFactory beanFactory new ConnectionDetailsFactories()); for (String beanName : beanFactory.getBeanNamesForType(Container.class)) { BeanDefinition beanDefinition = getBeanDefinition(beanFactory, beanName); - for (ServiceConnection annotation : getAnnotations(beanFactory, beanName, beanDefinition)) { - ContainerConnectionSource source = createSource(beanFactory, beanName, beanDefinition, annotation); + MergedAnnotations annotations = (beanDefinition instanceof TestcontainerBeanDefinition testcontainerBeanDefinition) + ? testcontainerBeanDefinition.getAnnotations() : null; + for (ServiceConnection serviceConnection : getServiceConnections(beanFactory, beanName, annotations)) { + ContainerConnectionSource source = createSource(beanFactory, beanName, beanDefinition, annotations, + serviceConnection); registrar.registerBeanDefinitions(registry, source); } } } - private Set getAnnotations(ConfigurableListableBeanFactory beanFactory, String beanName, - BeanDefinition beanDefinition) { - Set annotations = new LinkedHashSet<>( - beanFactory.findAllAnnotationsOnBean(beanName, ServiceConnection.class, false)); - if (beanDefinition instanceof TestcontainerBeanDefinition testcontainerBeanDefinition) { - testcontainerBeanDefinition.getAnnotations() - .stream(ServiceConnection.class) + private Set getServiceConnections(ConfigurableListableBeanFactory beanFactory, String beanName, + MergedAnnotations annotations) { + Set serviceConnections = beanFactory.findAllAnnotationsOnBean(beanName, + ServiceConnection.class, false); + if (annotations != null) { + serviceConnections = new LinkedHashSet<>(serviceConnections); + annotations.stream(ServiceConnection.class) .map(MergedAnnotation::synthesize) - .forEach(annotations::add); + .forEach(serviceConnections::add); } - return annotations; + return serviceConnections; } private BeanDefinition getBeanDefinition(ConfigurableListableBeanFactory beanFactory, String beanName) { @@ -91,13 +95,14 @@ private BeanDefinition getBeanDefinition(ConfigurableListableBeanFactory beanFac @SuppressWarnings("unchecked") private > ContainerConnectionSource createSource( ConfigurableListableBeanFactory beanFactory, String beanName, BeanDefinition beanDefinition, - ServiceConnection annotation) { + MergedAnnotations annotations, ServiceConnection serviceConnection) { Origin origin = new BeanOrigin(beanName, beanDefinition); Class containerType = (Class) beanFactory.getType(beanName, false); String containerImageName = (beanDefinition instanceof TestcontainerBeanDefinition testcontainerBeanDefinition) ? testcontainerBeanDefinition.getContainerImageName() : null; - return new ContainerConnectionSource<>(beanName, origin, containerType, containerImageName, annotation, - () -> beanFactory.getBean(beanName, containerType)); + return new ContainerConnectionSource<>(beanName, origin, containerType, containerImageName, serviceConnection, + () -> beanFactory.getBean(beanName, containerType), + SslBundleSource.get(beanFactory, beanName, annotations), annotations); } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizer.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizer.java index 9d20af2ac224..e5ade0d084ef 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizer.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -91,10 +91,12 @@ List> getSources() { * Relevant details from {@link ContainerConnectionSource} used as a * MergedContextConfiguration cache key. */ - private record CacheKey(String connectionName, Set> connectionDetailsTypes, Container container) { + private record CacheKey(String connectionName, Set> connectionDetailsTypes, Container container, + SslBundleSource sslBundleSource) { CacheKey(ContainerConnectionSource source) { - this(source.getConnectionName(), source.getConnectionDetailsTypes(), source.getContainerSupplier().get()); + this(source.getConnectionName(), source.getConnectionDetailsTypes(), source.getContainerSupplier().get(), + source.getSslBundleSource()); } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizerFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizerFactory.java index be70abf4527c..bbf6ac61b36d 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizerFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizerFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -61,7 +61,7 @@ private void collectSources(Class candidate, List { MergedAnnotations annotations = MergedAnnotations.from(field); annotations.stream(ServiceConnection.class) - .forEach((annotation) -> sources.add(createSource(field, annotation))); + .forEach((serviceConnection) -> sources.add(createSource(field, annotations, serviceConnection))); }); if (TestContextAnnotationUtils.searchEnclosingClass(candidate)) { collectSources(candidate.getEnclosingClass(), sources); @@ -74,7 +74,7 @@ private void collectSources(Class candidate, List> ContainerConnectionSource createSource(Field field, - MergedAnnotation annotation) { + MergedAnnotations annotations, MergedAnnotation serviceConnection) { Assert.state(Modifier.isStatic(field.getModifiers()), () -> "@ServiceConnection field '%s' must be static".formatted(field.getName())); Origin origin = new FieldOrigin(field); @@ -87,8 +87,8 @@ private > ContainerConnectionSource createSource(Field // When running tests that doesn't matter, but running AOT processing should be // possible without a Docker environment String dockerImageName = isAotProcessingInProgress() ? null : container.getDockerImageName(); - return new ContainerConnectionSource<>("test", origin, containerType, dockerImageName, annotation, - () -> container); + return new ContainerConnectionSource<>("test", origin, containerType, dockerImageName, serviceConnection, + () -> container, SslBundleSource.get(annotations), annotations); } private Object getFieldValue(Field field) { diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/Ssl.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/Ssl.java new file mode 100644 index 000000000000..51395569575b --- /dev/null +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/Ssl.java @@ -0,0 +1,78 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.testcontainers.service.connection; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslBundleKey; +import org.springframework.boot.ssl.SslOptions; + +/** + * Configures the {@link SslOptions}, {@link SslBundleKey @SslBundleKey} and + * {@link SslBundle#getProtocol() protocol} to use with an {@link SslBundle SSL} supported + * {@link ServiceConnection @ServiceConnection}. + *

+ * Also serves as a signal to enable automatic {@link SslBundle} extraction from supported + * containers. + * + * @author Phillip Webb + * @since 3.5.0 + */ +@Retention(RetentionPolicy.RUNTIME) +@Target({ ElementType.FIELD, ElementType.METHOD, ElementType.ANNOTATION_TYPE }) +public @interface Ssl { + + /** + * The protocol to use for the SSL connection. + * @return the SSL protocol + * @see SslBundle#getProtocol() + */ + String protocol() default SslBundle.DEFAULT_PROTOCOL; + + /** + * The ciphers that can be used for the SSL connection. + * @return the SSL ciphers + * @see SslOptions#getCiphers() + */ + String[] ciphers() default {}; + + /** + * The protocols that are enabled for the SSL connection. + * @return the enabled SSL protocols + * @see SslOptions#getEnabledProtocols() + */ + String[] enabledProtocols() default {}; + + /** + * The password that should be used to access the key. + * @return the key password + * @see SslBundleKey#getPassword() + */ + String keyPassword() default ""; + + /** + * The alias that should be used to access the key. + * @return the key alias + * @see SslBundleKey#getAlias() + */ + String keyAlias() default ""; + +} diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/SslBundleSource.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/SslBundleSource.java new file mode 100644 index 000000000000..2b32a7af7527 --- /dev/null +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/SslBundleSource.java @@ -0,0 +1,152 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.testcontainers.service.connection; + +import java.lang.annotation.Annotation; +import java.util.Collections; +import java.util.LinkedHashSet; +import java.util.Set; + +import org.springframework.beans.factory.ListableBeanFactory; +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslBundleKey; +import org.springframework.boot.ssl.SslOptions; +import org.springframework.boot.ssl.SslStoreBundle; +import org.springframework.boot.ssl.jks.JksSslStoreBundle; +import org.springframework.boot.ssl.jks.JksSslStoreDetails; +import org.springframework.boot.ssl.pem.PemSslStoreBundle; +import org.springframework.boot.ssl.pem.PemSslStoreDetails; +import org.springframework.core.annotation.MergedAnnotation; +import org.springframework.core.annotation.MergedAnnotations; +import org.springframework.util.Assert; +import org.springframework.util.StringUtils; + +/** + * {@link SslBundle} source created from annotations. Used as a cache key and as a + * {@link SslBundle} factory. + * + * @param ssl the {@link Ssl @Ssl} annotation + * @param pemKeyStore the {@link PemKeyStore @PemKeyStore} annotation + * @param pemTrustStore the {@link PemTrustStore @PemTrustStore} annotation + * @param jksKeyStore the {@link JksKeyStore @JksKeyStore} annotation + * @param jksTrustStore the {@link JksTrustStore @JksTrustStore} annotation + * @author Phillip Webb + * @author Moritz Halbritter + */ +record SslBundleSource(Ssl ssl, PemKeyStore pemKeyStore, PemTrustStore pemTrustStore, JksKeyStore jksKeyStore, + JksTrustStore jksTrustStore) { + + SslBundleSource { + boolean hasPem = (pemKeyStore != null || pemTrustStore != null); + boolean hasJks = (jksKeyStore != null || jksTrustStore != null); + if (hasJks && hasPem) { + throw new IllegalStateException("PEM and JKS store annotations cannot be used together"); + } + } + + SslBundle getSslBundle() { + SslStoreBundle stores = stores(); + if (stores == null) { + return null; + } + Ssl ssl = (this.ssl != null) ? this.ssl : MergedAnnotation.of(Ssl.class).synthesize(); + SslOptions options = SslOptions.of(nullIfEmpty(ssl.ciphers()), nullIfEmpty(ssl.enabledProtocols())); + SslBundleKey key = SslBundleKey.of(nullIfEmpty(ssl.keyPassword()), nullIfEmpty(ssl.keyAlias())); + String protocol = ssl.protocol(); + return SslBundle.of(stores, key, options, protocol); + } + + private SslStoreBundle stores() { + if (this.pemKeyStore != null || this.pemTrustStore != null) { + return new PemSslStoreBundle(pemKeyStoreDetails(), pemTrustStoreDetails()); + } + if (this.jksKeyStore != null || this.jksTrustStore != null) { + return new JksSslStoreBundle(jksKeyStoreDetails(), jksTrustStoreDetails()); + } + return null; + } + + private PemSslStoreDetails pemKeyStoreDetails() { + PemKeyStore store = this.pemKeyStore; + return (store != null) ? new PemSslStoreDetails(nullIfEmpty(store.type()), nullIfEmpty(store.certificate()), + nullIfEmpty(store.privateKey()), nullIfEmpty(store.privateKeyPassword())) : null; + } + + private PemSslStoreDetails pemTrustStoreDetails() { + PemTrustStore store = this.pemTrustStore; + return (store != null) ? new PemSslStoreDetails(nullIfEmpty(store.type()), nullIfEmpty(store.certificate()), + nullIfEmpty(store.privateKey()), nullIfEmpty(store.privateKeyPassword())) : null; + } + + private JksSslStoreDetails jksKeyStoreDetails() { + JksKeyStore store = this.jksKeyStore; + return (store != null) ? new JksSslStoreDetails(nullIfEmpty(store.type()), nullIfEmpty(store.provider()), + nullIfEmpty(store.location()), nullIfEmpty(store.password())) : null; + } + + private JksSslStoreDetails jksTrustStoreDetails() { + JksTrustStore store = this.jksTrustStore; + return (store != null) ? new JksSslStoreDetails(nullIfEmpty(store.type()), nullIfEmpty(store.provider()), + nullIfEmpty(store.location()), nullIfEmpty(store.password())) : null; + } + + private String nullIfEmpty(String string) { + if (StringUtils.hasLength(string)) { + return string; + } + return null; + } + + private String[] nullIfEmpty(String[] array) { + if (array == null || array.length == 0) { + return null; + } + return array; + } + + static SslBundleSource get(MergedAnnotations annotations) { + return get(null, null, annotations); + } + + static SslBundleSource get(ListableBeanFactory beanFactory, String beanName, MergedAnnotations annotations) { + Ssl ssl = getAnnotation(beanFactory, beanName, annotations, Ssl.class); + PemKeyStore pemKeyStore = getAnnotation(beanFactory, beanName, annotations, PemKeyStore.class); + PemTrustStore pemTrustStore = getAnnotation(beanFactory, beanName, annotations, PemTrustStore.class); + JksKeyStore jksKeyStore = getAnnotation(beanFactory, beanName, annotations, JksKeyStore.class); + JksTrustStore jksTrustStore = getAnnotation(beanFactory, beanName, annotations, JksTrustStore.class); + if (ssl == null && pemKeyStore == null && pemTrustStore == null && jksKeyStore == null + && jksTrustStore == null) { + return null; + } + return new SslBundleSource(ssl, pemKeyStore, pemTrustStore, jksKeyStore, jksTrustStore); + } + + private static A getAnnotation(ListableBeanFactory beanFactory, String beanName, + MergedAnnotations annotations, Class annotationType) { + Set found = (beanFactory != null) ? beanFactory.findAllAnnotationsOnBean(beanName, annotationType, false) + : Collections.emptySet(); + if (annotations != null) { + found = new LinkedHashSet<>(found); + annotations.stream(annotationType).map(MergedAnnotation::synthesize).forEach(found::add); + } + int size = found.size(); + Assert.state(size <= 1, + () -> "Expected single %s annotation, but found %d".formatted(annotationType.getName(), size)); + return (size > 0) ? found.iterator().next() : null; + } + +} diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/amqp/RabbitContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/amqp/RabbitContainerConnectionDetailsFactory.java index 6c92958428de..f665a5648f06 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/amqp/RabbitContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/amqp/RabbitContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,6 +22,7 @@ import org.testcontainers.containers.RabbitMQContainer; import org.springframework.boot.autoconfigure.amqp.RabbitConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; @@ -66,10 +67,15 @@ public String getPassword() { @Override public List

getAddresses() { - URI uri = URI.create(getContainer().getAmqpUrl()); + URI uri = URI.create((getSslBundle() != null) ? getContainer().getAmqpsUrl() : getContainer().getAmqpUrl()); return List.of(new Address(uri.getHost(), uri.getPort())); } + @Override + public SslBundle getSslBundle() { + return super.getSslBundle(); + } + } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/cassandra/CassandraContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/cassandra/CassandraContainerConnectionDetailsFactory.java index e8b82096ff07..f23cf7e3d58e 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/cassandra/CassandraContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/cassandra/CassandraContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,6 +22,7 @@ import org.testcontainers.cassandra.CassandraContainer; import org.springframework.boot.autoconfigure.cassandra.CassandraConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; @@ -75,6 +76,11 @@ public String getLocalDatacenter() { return getContainer().getLocalDatacenter(); } + @Override + public SslBundle getSslBundle() { + return super.getSslBundle(); + } + } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/cassandra/DeprecatedCassandraContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/cassandra/DeprecatedCassandraContainerConnectionDetailsFactory.java index 2ed24ac4444c..b1395cdfb2c7 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/cassandra/DeprecatedCassandraContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/cassandra/DeprecatedCassandraContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,6 +22,7 @@ import org.testcontainers.containers.CassandraContainer; import org.springframework.boot.autoconfigure.cassandra.CassandraConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; @@ -34,7 +35,7 @@ * @author Moritz Halbritter * @author Andy Wilkinson * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link CassandraContainerConnectionDetailsFactory}. */ @Deprecated(since = "3.4.0", forRemoval = true) @@ -78,6 +79,11 @@ public String getLocalDatacenter() { return getContainer().getLocalDatacenter(); } + @Override + public SslBundle getSslBundle() { + return super.getSslBundle(); + } + } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/couchbase/CouchbaseContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/couchbase/CouchbaseContainerConnectionDetailsFactory.java index 3d6ae0088b7e..cf67683815a7 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/couchbase/CouchbaseContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/couchbase/CouchbaseContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,7 @@ import org.testcontainers.couchbase.CouchbaseContainer; import org.springframework.boot.autoconfigure.couchbase.CouchbaseConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; @@ -66,6 +67,11 @@ public String getConnectionString() { return getContainer().getConnectionString(); } + @Override + public SslBundle getSslBundle() { + return super.getSslBundle(); + } + } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/elasticsearch/ElasticsearchContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/elasticsearch/ElasticsearchContainerConnectionDetailsFactory.java index eb2719a0262b..1f8d23c62ec0 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/elasticsearch/ElasticsearchContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/elasticsearch/ElasticsearchContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -16,15 +16,26 @@ package org.springframework.boot.testcontainers.service.connection.elasticsearch; +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.security.KeyStore; +import java.security.KeyStoreException; +import java.security.NoSuchAlgorithmException; +import java.security.cert.Certificate; +import java.security.cert.CertificateException; +import java.security.cert.CertificateFactory; import java.util.List; import org.testcontainers.elasticsearch.ElasticsearchContainer; import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchConnectionDetails; import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchConnectionDetails.Node.Protocol; +import org.springframework.boot.ssl.SslBundle; +import org.springframework.boot.ssl.SslStoreBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; +import org.springframework.boot.testcontainers.service.connection.Ssl; /** * {@link ContainerConnectionDetailsFactory} to create @@ -53,15 +64,83 @@ protected ElasticsearchConnectionDetails getContainerConnectionDetails( private static final class ElasticsearchContainerConnectionDetails extends ContainerConnectionDetails implements ElasticsearchConnectionDetails { + private volatile SslBundle sslBundle; + private ElasticsearchContainerConnectionDetails(ContainerConnectionSource source) { super(source); } + @Override + public String getUsername() { + return "elastic"; + } + + @Override + public String getPassword() { + return getContainer().getEnvMap().get("ELASTIC_PASSWORD"); + } + @Override public List getNodes() { String host = getContainer().getHost(); Integer port = getContainer().getMappedPort(DEFAULT_PORT); - return List.of(new Node(host, port, Protocol.HTTP, null, null)); + return List.of(new Node(host, port, (getSslBundle() != null) ? Protocol.HTTPS : Protocol.HTTP, + getUsername(), getPassword())); + } + + @Override + public SslBundle getSslBundle() { + if (this.sslBundle != null) { + return this.sslBundle; + } + SslBundle sslBundle = super.getSslBundle(); + if (sslBundle != null) { + this.sslBundle = sslBundle; + return sslBundle; + } + if (hasAnnotation(Ssl.class)) { + byte[] caCertificate = getContainer().caCertAsBytes().orElse(null); + if (caCertificate != null) { + KeyStore trustStore = createTrustStore(caCertificate); + sslBundle = createSslBundleWithTrustStore(trustStore); + this.sslBundle = sslBundle; + return sslBundle; + } + } + return null; + } + + private SslBundle createSslBundleWithTrustStore(KeyStore trustStore) { + return SslBundle.of(new SslStoreBundle() { + @Override + public KeyStore getKeyStore() { + return null; + } + + @Override + public String getKeyStorePassword() { + return null; + } + + @Override + public KeyStore getTrustStore() { + return trustStore; + } + }); + } + + private KeyStore createTrustStore(byte[] caCertificate) { + try { + KeyStore keyStore = KeyStore.getInstance(KeyStore.getDefaultType()); + keyStore.load(null, null); + CertificateFactory certFactory = CertificateFactory.getInstance("X.509"); + Certificate certificate = certFactory.generateCertificate(new ByteArrayInputStream(caCertificate)); + keyStore.setCertificateEntry("ca", certificate); + return keyStore; + } + catch (KeyStoreException | CertificateException | IOException | NoSuchAlgorithmException ex) { + throw new IllegalStateException("Failed to create keystore from CA certificate", ex); + } } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/ApacheKafkaContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/ApacheKafkaContainerConnectionDetailsFactory.java index 5815e75ac82a..64deae99cd22 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/ApacheKafkaContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/ApacheKafkaContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,6 +21,7 @@ import org.testcontainers.kafka.KafkaContainer; import org.springframework.boot.autoconfigure.kafka.KafkaConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; @@ -57,6 +58,16 @@ public List getBootstrapServers() { return List.of(getContainer().getBootstrapServers()); } + @Override + public SslBundle getSslBundle() { + return super.getSslBundle(); + } + + @Override + public String getSecurityProtocol() { + return (getSslBundle() != null) ? "SSL" : "PLAINTEXT"; + } + } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/ConfluentKafkaContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/ConfluentKafkaContainerConnectionDetailsFactory.java index d7a23db29572..a102b3cf50ba 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/ConfluentKafkaContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/ConfluentKafkaContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,6 +21,7 @@ import org.testcontainers.kafka.ConfluentKafkaContainer; import org.springframework.boot.autoconfigure.kafka.KafkaConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; @@ -58,6 +59,16 @@ public List getBootstrapServers() { return List.of(getContainer().getBootstrapServers()); } + @Override + public SslBundle getSslBundle() { + return super.getSslBundle(); + } + + @Override + public String getSecurityProtocol() { + return (getSslBundle() != null) ? "SSL" : "PLAINTEXT"; + } + } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/DeprecatedConfluentKafkaContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/DeprecatedConfluentKafkaContainerConnectionDetailsFactory.java index eafd40c87efa..3a31c5227619 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/DeprecatedConfluentKafkaContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/kafka/DeprecatedConfluentKafkaContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,6 +21,7 @@ import org.testcontainers.containers.KafkaContainer; import org.springframework.boot.autoconfigure.kafka.KafkaConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; @@ -32,7 +33,7 @@ * @author Moritz Halbritter * @author Andy Wilkinson * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link ConfluentKafkaContainerConnectionDetailsFactory}. */ @Deprecated(since = "3.4.0", forRemoval = true) @@ -59,6 +60,16 @@ public List getBootstrapServers() { return List.of(getContainer().getBootstrapServers()); } + @Override + public SslBundle getSslBundle() { + return super.getSslBundle(); + } + + @Override + public String getSecurityProtocol() { + return (getSslBundle() != null) ? "SSL" : "PLAINTEXT"; + } + } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ldap/LLdapContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ldap/LLdapContainerConnectionDetailsFactory.java new file mode 100644 index 000000000000..6b419a9d0921 --- /dev/null +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/ldap/LLdapContainerConnectionDetailsFactory.java @@ -0,0 +1,69 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.testcontainers.service.connection.ldap; + +import org.testcontainers.ldap.LLdapContainer; + +import org.springframework.boot.autoconfigure.ldap.LdapConnectionDetails; +import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; +import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; +import org.springframework.boot.testcontainers.service.connection.ServiceConnection; + +/** + * {@link ContainerConnectionDetailsFactory} to create {@link LdapConnectionDetails} from + * a {@link ServiceConnection @ServiceConnection}-annotated {@link LLdapContainer}. + * + * @author Eddú Meléndez + */ +class LLdapContainerConnectionDetailsFactory + extends ContainerConnectionDetailsFactory { + + @Override + protected LdapConnectionDetails getContainerConnectionDetails(ContainerConnectionSource source) { + return new LLdapContainerConnectionDetails(source); + } + + private static final class LLdapContainerConnectionDetails extends ContainerConnectionDetails + implements LdapConnectionDetails { + + private LLdapContainerConnectionDetails(ContainerConnectionSource source) { + super(source); + } + + @Override + public String[] getUrls() { + return new String[] { getContainer().getLdapUrl() }; + } + + @Override + public String getBase() { + return getContainer().getBaseDn(); + } + + @Override + public String getUsername() { + return getContainer().getUser(); + } + + @Override + public String getPassword() { + return getContainer().getUserPass(); + } + + } + +} diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/mongo/MongoContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/mongo/MongoContainerConnectionDetailsFactory.java index 619f42b271f6..90173be4efc4 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/mongo/MongoContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/mongo/MongoContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,6 +20,7 @@ import org.testcontainers.containers.MongoDBContainer; import org.springframework.boot.autoconfigure.mongo.MongoConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; @@ -59,6 +60,11 @@ public ConnectionString getConnectionString() { return new ConnectionString(getContainer().getReplicaSetUrl()); } + @Override + public SslBundle getSslBundle() { + return super.getSslBundle(); + } + } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/redis/RedisContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/redis/RedisContainerConnectionDetailsFactory.java index 134824058dc2..ea2095cb3b0f 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/redis/RedisContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/redis/RedisContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -75,7 +75,8 @@ private RedisContainerConnectionDetails(ContainerConnectionSource> @Override public Standalone getStandalone() { - return Standalone.of(getContainer().getHost(), getContainer().getMappedPort(REDIS_PORT)); + return Standalone.of(getContainer().getHost(), getContainer().getMappedPort(REDIS_PORT), + super.getSslBundle()); } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/redpanda/RedpandaContainerConnectionDetailsFactory.java b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/redpanda/RedpandaContainerConnectionDetailsFactory.java index 48ec53bec11f..86693d18f045 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/redpanda/RedpandaContainerConnectionDetailsFactory.java +++ b/spring-boot-project/spring-boot-testcontainers/src/main/java/org/springframework/boot/testcontainers/service/connection/redpanda/RedpandaContainerConnectionDetailsFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,6 +21,7 @@ import org.testcontainers.redpanda.RedpandaContainer; import org.springframework.boot.autoconfigure.kafka.KafkaConnectionDetails; +import org.springframework.boot.ssl.SslBundle; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionDetailsFactory; import org.springframework.boot.testcontainers.service.connection.ContainerConnectionSource; import org.springframework.boot.testcontainers.service.connection.ServiceConnection; @@ -55,6 +56,16 @@ public List getBootstrapServers() { return List.of(getContainer().getBootstrapServers()); } + @Override + public SslBundle getSslBundle() { + return super.getSslBundle(); + } + + @Override + public String getSecurityProtocol() { + return (getSslBundle() != null) ? "SSL" : "PLAINTEXT"; + } + } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/main/resources/META-INF/spring.factories b/spring-boot-project/spring-boot-testcontainers/src/main/resources/META-INF/spring.factories index 902b2c43aca2..20a62beb5fbe 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/main/resources/META-INF/spring.factories +++ b/spring-boot-project/spring-boot-testcontainers/src/main/resources/META-INF/spring.factories @@ -22,6 +22,7 @@ org.springframework.boot.testcontainers.service.connection.jdbc.JdbcContainerCon org.springframework.boot.testcontainers.service.connection.kafka.ApacheKafkaContainerConnectionDetailsFactory,\ org.springframework.boot.testcontainers.service.connection.kafka.ConfluentKafkaContainerConnectionDetailsFactory,\ org.springframework.boot.testcontainers.service.connection.kafka.DeprecatedConfluentKafkaContainerConnectionDetailsFactory,\ +org.springframework.boot.testcontainers.service.connection.ldap.LLdapContainerConnectionDetailsFactory,\ org.springframework.boot.testcontainers.service.connection.ldap.OpenLdapContainerConnectionDetailsFactory,\ org.springframework.boot.testcontainers.service.connection.liquibase.LiquibaseContainerConnectionDetailsFactory,\ org.springframework.boot.testcontainers.service.connection.mongo.MongoContainerConnectionDetailsFactory,\ diff --git a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/properties/TestcontainersPropertySourceTests.java b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/properties/TestcontainersPropertySourceTests.java index 177322aeedec..2621b47df13f 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/properties/TestcontainersPropertySourceTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/properties/TestcontainersPropertySourceTests.java @@ -41,7 +41,7 @@ * Tests for {@link TestcontainersPropertySource}. * * @author Phillip Webb - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @SuppressWarnings("removal") @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ConnectionDetailsRegistrarTests.java b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ConnectionDetailsRegistrarTests.java index d40ce9664c35..179931ed8b99 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ConnectionDetailsRegistrarTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ConnectionDetailsRegistrarTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -59,7 +59,7 @@ void setup() { this.container = mock(PostgreSQLContainer.class); this.annotation = MergedAnnotation.of(ServiceConnection.class, Map.of("name", "", "type", new Class[0])); this.source = new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, null, - this.annotation, () -> this.container); + this.annotation, () -> this.container, null, null); this.factories = mock(ConnectionDetailsFactories.class); } diff --git a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionDetailsFactoryTests.java b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionDetailsFactoryTests.java index b24b25f1a810..828cec1e5ea0 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionDetailsFactoryTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionDetailsFactoryTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -65,7 +65,7 @@ void setup() { this.annotation = MergedAnnotation.of(ServiceConnection.class, Map.of("name", "myname", "type", new Class[0])); this.source = new ContainerConnectionSource<>(this.beanNameSuffix, this.origin, PostgreSQLContainer.class, - this.container.getDockerImageName(), this.annotation, () -> this.container); + this.container.getDockerImageName(), this.annotation, () -> this.container, null, null); } @Test @@ -109,7 +109,8 @@ void getConnectionDetailsWhenTypesMatchAndNameRestrictionsDoNotMatchReturnsNull( void getConnectionDetailsWhenContainerTypeDoesNotMatchReturnsNull() { ElasticsearchContainer container = mock(ElasticsearchContainer.class); ContainerConnectionSource source = new ContainerConnectionSource<>(this.beanNameSuffix, this.origin, - ElasticsearchContainer.class, container.getDockerImageName(), this.annotation, () -> container); + ElasticsearchContainer.class, container.getDockerImageName(), this.annotation, () -> container, null, + null); TestContainerConnectionDetailsFactory factory = new TestContainerConnectionDetailsFactory(); ConnectionDetails connectionDetails = getConnectionDetails(factory, source); assertThat(connectionDetails).isNull(); diff --git a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionSourceTests.java b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionSourceTests.java index 913b08037ed9..fa22ec512d8e 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionSourceTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ContainerConnectionSourceTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -60,7 +60,7 @@ void setup() { given(this.container.getDockerImageName()).willReturn("postgres"); this.annotation = MergedAnnotation.of(ServiceConnection.class, Map.of("name", "", "type", new Class[0])); this.source = new ContainerConnectionSource<>(this.beanNameSuffix, this.origin, PostgreSQLContainer.class, - this.container.getDockerImageName(), this.annotation, () -> this.container); + this.container.getDockerImageName(), this.annotation, () -> this.container, null, null); } @Test @@ -180,14 +180,14 @@ void toStringReturnsSensibleString() { private void setupSourceAnnotatedWithName(String name) { this.annotation = MergedAnnotation.of(ServiceConnection.class, Map.of("name", name, "type", new Class[0])); this.source = new ContainerConnectionSource<>(this.beanNameSuffix, this.origin, PostgreSQLContainer.class, - this.container.getDockerImageName(), this.annotation, () -> this.container); + this.container.getDockerImageName(), this.annotation, () -> this.container, null, null); } private void setupSourceAnnotatedWithType(Class type) { this.annotation = MergedAnnotation.of(ServiceConnection.class, Map.of("name", "", "type", new Class[] { type })); this.source = new ContainerConnectionSource<>(this.beanNameSuffix, this.origin, PostgreSQLContainer.class, - this.container.getDockerImageName(), this.annotation, () -> this.container); + this.container.getDockerImageName(), this.annotation, () -> this.container, null, null); } } diff --git a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/FieldOriginTests.java b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/FieldOriginTests.java index d3b89a8d6145..c0e0a69ded27 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/FieldOriginTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/FieldOriginTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,7 +36,7 @@ class FieldOriginTests { @Test void createWhenFieldIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new FieldOrigin(null)) - .withMessage("Field must not be null"); + .withMessage("'field' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizerTests.java b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizerTests.java index 9ce9c30f9b05..04a084f2448d 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizerTests.java +++ b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/ServiceConnectionContextCustomizerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -66,7 +66,7 @@ void setup() { this.annotation = MergedAnnotation.of(ServiceConnection.class, Map.of("name", "myname", "type", new Class[0])); this.source = new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, - this.container.getDockerImageName(), this.annotation, () -> this.container); + this.container.getDockerImageName(), this.annotation, () -> this.container, null, null); this.factories = mock(ConnectionDetailsFactories.class); } @@ -103,37 +103,37 @@ void equalsAndHashCode() { // Connection Names ServiceConnectionContextCustomizer n1 = new ServiceConnectionContextCustomizer( List.of(new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, "name", - annotation1, () -> container1))); + annotation1, () -> container1, null, null))); ServiceConnectionContextCustomizer n2 = new ServiceConnectionContextCustomizer( List.of(new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, "name", - annotation1, () -> container1))); + annotation1, () -> container1, null, null))); ServiceConnectionContextCustomizer n3 = new ServiceConnectionContextCustomizer( List.of(new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, "namex", - annotation1, () -> container1))); + annotation1, () -> container1, null, null))); assertThat(n1.hashCode()).isEqualTo(n2.hashCode()).isNotEqualTo(n3.hashCode()); assertThat(n1).isEqualTo(n2).isNotEqualTo(n3); // Connection Details Types ServiceConnectionContextCustomizer t1 = new ServiceConnectionContextCustomizer( List.of(new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, "name", - annotation1, () -> container1))); + annotation1, () -> container1, null, null))); ServiceConnectionContextCustomizer t2 = new ServiceConnectionContextCustomizer( List.of(new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, "name", - annotation2, () -> container1))); + annotation2, () -> container1, null, null))); ServiceConnectionContextCustomizer t3 = new ServiceConnectionContextCustomizer( List.of(new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, "name", - annotation3, () -> container1))); + annotation3, () -> container1, null, null))); assertThat(t1.hashCode()).isEqualTo(t2.hashCode()).isNotEqualTo(t3.hashCode()); assertThat(t1).isEqualTo(t2).isNotEqualTo(t3); // Container ServiceConnectionContextCustomizer c1 = new ServiceConnectionContextCustomizer( List.of(new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, "name", - annotation1, () -> container1))); + annotation1, () -> container1, null, null))); ServiceConnectionContextCustomizer c2 = new ServiceConnectionContextCustomizer( List.of(new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, "name", - annotation1, () -> container1))); + annotation1, () -> container1, null, null))); ServiceConnectionContextCustomizer c3 = new ServiceConnectionContextCustomizer( List.of(new ContainerConnectionSource<>("test", this.origin, PostgreSQLContainer.class, "name", - annotation1, () -> container2))); + annotation1, () -> container2, null, null))); assertThat(c1.hashCode()).isEqualTo(c2.hashCode()).isNotEqualTo(c3.hashCode()); assertThat(c1).isEqualTo(c2).isNotEqualTo(c3); } diff --git a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/TestContainerConnectionSource.java b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/TestContainerConnectionSource.java index dda8088eceb6..55515a134cc1 100644 --- a/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/TestContainerConnectionSource.java +++ b/spring-boot-project/spring-boot-testcontainers/src/test/java/org/springframework/boot/testcontainers/service/connection/TestContainerConnectionSource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ public static > ContainerConnectionSource create(Strin Class containerType, String containerImageName, MergedAnnotation annotation, Supplier containerSupplier) { return new ContainerConnectionSource<>(beanNameSuffix, origin, containerType, containerImageName, annotation, - containerSupplier); + containerSupplier, null, null); } } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildOwner.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildOwner.java index a027d7237cd0..077cd9eb7b19 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildOwner.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildOwner.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -84,7 +84,7 @@ public String toString() { * @throws IllegalStateException if the env does not contain the correct CNB variables */ static BuildOwner fromEnv(Map env) { - Assert.notNull(env, "Env must not be null"); + Assert.notNull(env, "'env' must not be null"); return new BuildOwner(env); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildRequest.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildRequest.java index f17e717249fd..2011bf6a9a8d 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildRequest.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildRequest.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -108,8 +108,8 @@ public class BuildRequest { private final ImagePlatform platform; BuildRequest(ImageReference name, Function applicationContent) { - Assert.notNull(name, "Name must not be null"); - Assert.notNull(applicationContent, "ApplicationContent must not be null"); + Assert.notNull(name, "'name' must not be null"); + Assert.notNull(applicationContent, "'applicationContent' must not be null"); this.name = name.inTaggedForm(); this.applicationContent = applicationContent; this.builder = DEFAULT_BUILDER; @@ -170,7 +170,7 @@ public class BuildRequest { * @return an updated build request */ public BuildRequest withBuilder(ImageReference builder) { - Assert.notNull(builder, "Builder must not be null"); + Assert.notNull(builder, "'builder' must not be null"); return new BuildRequest(this.name, this.applicationContent, builder.inTaggedOrDigestForm(), this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, this.tags, this.buildWorkspace, @@ -211,7 +211,7 @@ public BuildRequest withRunImage(ImageReference runImageName) { * @return an updated build request */ public BuildRequest withCreator(Creator creator) { - Assert.notNull(creator, "Creator must not be null"); + Assert.notNull(creator, "'creator' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, this.tags, this.buildWorkspace, this.buildCache, this.launchCache, @@ -225,8 +225,8 @@ public BuildRequest withCreator(Creator creator) { * @return an updated build request */ public BuildRequest withEnv(String name, String value) { - Assert.hasText(name, "Name must not be empty"); - Assert.hasText(value, "Value must not be empty"); + Assert.hasText(name, "'name' must not be empty"); + Assert.hasText(value, "'value' must not be empty"); Map env = new LinkedHashMap<>(this.env); env.put(name, value); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, @@ -242,7 +242,7 @@ public BuildRequest withEnv(String name, String value) { * @return an updated build request */ public BuildRequest withEnv(Map env) { - Assert.notNull(env, "Env must not be null"); + Assert.notNull(env, "'env' must not be null"); Map updatedEnv = new LinkedHashMap<>(this.env); updatedEnv.putAll(env); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, @@ -307,7 +307,7 @@ public BuildRequest withPublish(boolean publish) { * @since 2.5.0 */ public BuildRequest withBuildpacks(BuildpackReference... buildpacks) { - Assert.notEmpty(buildpacks, "Buildpacks must not be empty"); + Assert.notEmpty(buildpacks, "'buildpacks' must not be empty"); return withBuildpacks(Arrays.asList(buildpacks)); } @@ -318,7 +318,7 @@ public BuildRequest withBuildpacks(BuildpackReference... buildpacks) { * @since 2.5.0 */ public BuildRequest withBuildpacks(List buildpacks) { - Assert.notNull(buildpacks, "Buildpacks must not be null"); + Assert.notNull(buildpacks, "'buildpacks' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, buildpacks, this.bindings, this.network, this.tags, this.buildWorkspace, this.buildCache, this.launchCache, @@ -332,7 +332,7 @@ public BuildRequest withBuildpacks(List buildpacks) { * @since 2.5.0 */ public BuildRequest withBindings(Binding... bindings) { - Assert.notEmpty(bindings, "Bindings must not be empty"); + Assert.notEmpty(bindings, "'bindings' must not be empty"); return withBindings(Arrays.asList(bindings)); } @@ -343,7 +343,7 @@ public BuildRequest withBindings(Binding... bindings) { * @since 2.5.0 */ public BuildRequest withBindings(List bindings) { - Assert.notNull(bindings, "Bindings must not be null"); + Assert.notNull(bindings, "'bindings' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, bindings, this.network, this.tags, this.buildWorkspace, this.buildCache, @@ -369,7 +369,7 @@ public BuildRequest withNetwork(String network) { * @return an updated build request */ public BuildRequest withTags(ImageReference... tags) { - Assert.notEmpty(tags, "Tags must not be empty"); + Assert.notEmpty(tags, "'tags' must not be empty"); return withTags(Arrays.asList(tags)); } @@ -379,7 +379,7 @@ public BuildRequest withTags(ImageReference... tags) { * @return an updated build request */ public BuildRequest withTags(List tags) { - Assert.notNull(tags, "Tags must not be null"); + Assert.notNull(tags, "'tags' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, tags, this.buildWorkspace, this.buildCache, @@ -393,7 +393,7 @@ public BuildRequest withTags(List tags) { * @since 3.2.0 */ public BuildRequest withBuildWorkspace(Cache buildWorkspace) { - Assert.notNull(buildWorkspace, "BuildWorkspace must not be null"); + Assert.notNull(buildWorkspace, "'buildWorkspace' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, this.tags, buildWorkspace, this.buildCache, @@ -406,7 +406,7 @@ public BuildRequest withBuildWorkspace(Cache buildWorkspace) { * @return an updated build request */ public BuildRequest withBuildCache(Cache buildCache) { - Assert.notNull(buildCache, "BuildCache must not be null"); + Assert.notNull(buildCache, "'buildCache' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, this.tags, this.buildWorkspace, buildCache, @@ -419,7 +419,7 @@ public BuildRequest withBuildCache(Cache buildCache) { * @return an updated build request */ public BuildRequest withLaunchCache(Cache launchCache) { - Assert.notNull(launchCache, "LaunchCache must not be null"); + Assert.notNull(launchCache, "'launchCache' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, this.tags, this.buildWorkspace, this.buildCache, @@ -432,7 +432,7 @@ public BuildRequest withLaunchCache(Cache launchCache) { * @return an updated build request */ public BuildRequest withCreatedDate(String createdDate) { - Assert.notNull(createdDate, "CreatedDate must not be null"); + Assert.notNull(createdDate, "'createdDate' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, this.tags, this.buildWorkspace, this.buildCache, @@ -458,7 +458,7 @@ private Instant parseCreatedDate(String createdDate) { * @return an updated build request */ public BuildRequest withApplicationDirectory(String applicationDirectory) { - Assert.notNull(applicationDirectory, "ApplicationDirectory must not be null"); + Assert.notNull(applicationDirectory, "'applicationDirectory' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, this.tags, this.buildWorkspace, this.buildCache, @@ -472,7 +472,7 @@ public BuildRequest withApplicationDirectory(String applicationDirectory) { * @since 3.2.0 */ public BuildRequest withSecurityOptions(List securityOptions) { - Assert.notNull(securityOptions, "SecurityOption must not be null"); + Assert.notNull(securityOptions, "'securityOptions' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, this.tags, this.buildWorkspace, this.buildCache, @@ -486,7 +486,7 @@ public BuildRequest withSecurityOptions(List securityOptions) { * @since 3.4.0 */ public BuildRequest withImagePlatform(String platform) { - Assert.notNull(platform, "Platform must not be null"); + Assert.notNull(platform, "'platform' must not be null"); return new BuildRequest(this.name, this.applicationContent, this.builder, this.trustBuilder, this.runImage, this.creator, this.env, this.cleanCache, this.verboseLogging, this.pullPolicy, this.publish, this.buildpacks, this.bindings, this.network, this.tags, this.buildWorkspace, this.buildCache, @@ -715,9 +715,9 @@ public static BuildRequest of(ImageReference name, Function a } private static void assertJarFile(File jarFile) { - Assert.notNull(jarFile, "JarFile must not be null"); - Assert.isTrue(jarFile.exists(), "JarFile must exist"); - Assert.isTrue(jarFile.isFile(), "JarFile must be a file"); + Assert.notNull(jarFile, "'jarFile' must not be null"); + Assert.isTrue(jarFile.exists(), "'jarFile' must exist"); + Assert.isTrue(jarFile.isFile(), "'jarFile' must be a file"); } } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/Builder.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/Builder.java index ebbca2b0534b..95764b42f641 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/Builder.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/Builder.java @@ -21,6 +21,7 @@ import java.util.function.Consumer; import org.springframework.boot.buildpack.platform.docker.DockerApi; +import org.springframework.boot.buildpack.platform.docker.DockerLog; import org.springframework.boot.buildpack.platform.docker.TotalProgressEvent; import org.springframework.boot.buildpack.platform.docker.TotalProgressPullListener; import org.springframework.boot.buildpack.platform.docker.TotalProgressPushListener; @@ -75,7 +76,7 @@ public Builder(DockerConfiguration dockerConfiguration) { * @param log a logger used to record output */ public Builder(BuildLog log) { - this(log, new DockerApi(), null); + this(log, new DockerApi(null, BuildLogAdapter.get(log)), null); } /** @@ -85,19 +86,19 @@ public Builder(BuildLog log) { * @since 2.4.0 */ public Builder(BuildLog log, DockerConfiguration dockerConfiguration) { - this(log, new DockerApi((dockerConfiguration != null) ? dockerConfiguration.getHost() : null), - dockerConfiguration); + this(log, new DockerApi((dockerConfiguration != null) ? dockerConfiguration.getHost() : null, + BuildLogAdapter.get(log)), dockerConfiguration); } Builder(BuildLog log, DockerApi docker, DockerConfiguration dockerConfiguration) { - Assert.notNull(log, "Log must not be null"); + Assert.notNull(log, "'log' must not be null"); this.log = log; this.docker = docker; this.dockerConfiguration = dockerConfiguration; } public void build(BuildRequest request) throws DockerEngineException, IOException { - Assert.notNull(request, "Request must not be null"); + Assert.notNull(request, "'request' must not be null"); this.log.start(request); validateBindings(request.getBindings()); String domain = request.getBuilder().getDomain(); @@ -230,8 +231,8 @@ private class ImageFetcher { } Image fetchImage(ImageType type, ImageReference reference) throws IOException { - Assert.notNull(type, "Type must not be null"); - Assert.notNull(reference, "Reference must not be null"); + Assert.notNull(type, "'type' must not be null"); + Assert.notNull(reference, "'reference' must not be null"); Assert.state(this.authHeader == null || reference.getDomain().equals(this.domain), () -> String.format("%s '%s' must be pulled from the '%s' authenticated registry", StringUtils.capitalize(type.getDescription()), reference, this.domain)); @@ -282,6 +283,40 @@ private PlatformMismatchException(ImageReference imageReference, ImagePlatform r } + /** + * A {@link DockerLog} implementation that adapts to an {@link AbstractBuildLog}. + */ + static final class BuildLogAdapter implements DockerLog { + + private final AbstractBuildLog log; + + private BuildLogAdapter(AbstractBuildLog log) { + this.log = log; + } + + @Override + public void log(String message) { + this.log.log(message); + } + + /** + * Creates {@link DockerLog} instance based on the provided {@link BuildLog}. + *

+ * If the provided {@link BuildLog} instance is an {@link AbstractBuildLog}, the + * method returns a {@link BuildLogAdapter}, otherwise it returns a default + * {@link DockerLog#toSystemOut()}. + * @param log the {@link BuildLog} instance to delegate + * @return a {@link DockerLog} instance for logging + */ + static DockerLog get(BuildLog log) { + if (log instanceof AbstractBuildLog abstractBuildLog) { + return new BuildLogAdapter(abstractBuildLog); + } + return DockerLog.toSystemOut(); + } + + } + /** * {@link BuildpackResolverContext} implementation for the {@link Builder}. */ diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuilderBuildpack.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuilderBuildpack.java index e649586e18b5..aada48b018c3 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuilderBuildpack.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuilderBuildpack.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -65,7 +65,7 @@ static Buildpack resolve(BuildpackResolverContext context, BuildpackReference re .of(unambiguous ? reference.getSubReference(PREFIX) : reference.toString()); BuildpackMetadata buildpackMetadata = findBuildpackMetadata(context, builderReference); if (unambiguous) { - Assert.isTrue(buildpackMetadata != null, () -> "Buildpack '" + reference + "' not found in builder"); + Assert.state(buildpackMetadata != null, () -> "Buildpack '" + reference + "' not found in builder"); } return (buildpackMetadata != null) ? new BuilderBuildpack(buildpackMetadata) : null; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuilderMetadata.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuilderMetadata.java index 831b524a3c83..2d5a54bb631e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuilderMetadata.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuilderMetadata.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -145,7 +145,7 @@ void attachTo(ImageConfig.Update update) { * @throws IOException on IO error */ static BuilderMetadata fromImage(Image image) throws IOException { - Assert.notNull(image, "Image must not be null"); + Assert.notNull(image, "'image' must not be null"); return fromImageConfig(image.getConfig()); } @@ -156,9 +156,9 @@ static BuilderMetadata fromImage(Image image) throws IOException { * @throws IOException on IO error */ static BuilderMetadata fromImageConfig(ImageConfig imageConfig) throws IOException { - Assert.notNull(imageConfig, "ImageConfig must not be null"); + Assert.notNull(imageConfig, "'imageConfig' must not be null"); String json = imageConfig.getLabels().get(LABEL_NAME); - Assert.notNull(json, () -> "No '" + LABEL_NAME + "' label found in image config labels '" + Assert.state(json != null, () -> "No '" + LABEL_NAME + "' label found in image config labels '" + StringUtils.collectionToCommaDelimitedString(imageConfig.getLabels().keySet()) + "'"); return fromJson(json); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackCoordinates.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackCoordinates.java index 954ddd902589..fd667cd68b1e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackCoordinates.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackCoordinates.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,7 +42,7 @@ final class BuildpackCoordinates { private final String version; private BuildpackCoordinates(String id, String version) { - Assert.hasText(id, "ID must not be empty"); + Assert.hasText(id, "'id' must not be empty"); this.id = id; this.version = version; } @@ -123,7 +123,7 @@ private static BuildpackCoordinates fromToml(TomlParseResult toml, Path path) { * @return a new {@link BuildpackCoordinates} instance */ static BuildpackCoordinates fromBuildpackMetadata(BuildpackMetadata buildpackMetadata) { - Assert.notNull(buildpackMetadata, "BuildpackMetadata must not be null"); + Assert.notNull(buildpackMetadata, "'buildpackMetadata' must not be null"); return new BuildpackCoordinates(buildpackMetadata.getId(), buildpackMetadata.getVersion()); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackLayersMetadata.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackLayersMetadata.java index 689729fa89c1..ac9f54591e9c 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackLayersMetadata.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackLayersMetadata.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -64,7 +64,7 @@ BuildpackLayerDetails getBuildpack(String id, String version) { * @throws IOException on IO error */ static BuildpackLayersMetadata fromImage(Image image) throws IOException { - Assert.notNull(image, "Image must not be null"); + Assert.notNull(image, "'image' must not be null"); return fromImageConfig(image.getConfig()); } @@ -75,9 +75,9 @@ static BuildpackLayersMetadata fromImage(Image image) throws IOException { * @throws IOException on IO error */ static BuildpackLayersMetadata fromImageConfig(ImageConfig imageConfig) throws IOException { - Assert.notNull(imageConfig, "ImageConfig must not be null"); + Assert.notNull(imageConfig, "'imageConfig' must not be null"); String json = imageConfig.getLabels().get(LABEL_NAME); - Assert.notNull(json, () -> "No '" + LABEL_NAME + "' label found in image config labels '" + Assert.state(json != null, () -> "No '" + LABEL_NAME + "' label found in image config labels '" + StringUtils.collectionToCommaDelimitedString(imageConfig.getLabels().keySet()) + "'"); return fromJson(json); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackMetadata.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackMetadata.java index 5b3d1e109860..79063567f0e6 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackMetadata.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackMetadata.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -81,7 +81,7 @@ String getHomepage() { * @throws IOException on IO error */ static BuildpackMetadata fromImage(Image image) throws IOException { - Assert.notNull(image, "Image must not be null"); + Assert.notNull(image, "'image' must not be null"); return fromImageConfig(image.getConfig()); } @@ -92,9 +92,9 @@ static BuildpackMetadata fromImage(Image image) throws IOException { * @throws IOException on IO error */ static BuildpackMetadata fromImageConfig(ImageConfig imageConfig) throws IOException { - Assert.notNull(imageConfig, "ImageConfig must not be null"); + Assert.notNull(imageConfig, "'imageConfig' must not be null"); String json = imageConfig.getLabels().get(LABEL_NAME); - Assert.notNull(json, () -> "No '" + LABEL_NAME + "' label found in image config labels '" + Assert.state(json != null, () -> "No '" + LABEL_NAME + "' label found in image config labels '" + StringUtils.collectionToCommaDelimitedString(imageConfig.getLabels().keySet()) + "'"); return fromJson(json); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackReference.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackReference.java index ebbf7da13277..a555672b9c59 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackReference.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackReference.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -94,7 +94,7 @@ public String toString() { * @return a new {@link BuildpackReference} */ public static BuildpackReference of(String value) { - Assert.hasText(value, "Value must not be empty"); + Assert.hasText(value, "'value' must not be empty"); return new BuildpackReference(value); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackResolvers.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackResolvers.java index 1883df4264cd..ec64d0a9b590 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackResolvers.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/BuildpackResolvers.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -55,7 +55,7 @@ private static List getResolvers() { * @return a {@link Buildpacks} instance */ static Buildpacks resolveAll(BuildpackResolverContext context, Collection references) { - Assert.notNull(context, "Context must not be null"); + Assert.notNull(context, "'context' must not be null"); if (CollectionUtils.isEmpty(references)) { return Buildpacks.EMPTY; } @@ -67,7 +67,7 @@ static Buildpacks resolveAll(BuildpackResolverContext context, Collection "Buildpack descriptor 'buildpack.toml' is required in buildpack '" + path + "'"); try { try (InputStream inputStream = Files.newInputStream(buildpackToml)) { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/LifecycleVersion.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/LifecycleVersion.java index 3db3f48f66c7..9936c6323b6c 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/LifecycleVersion.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/LifecycleVersion.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -120,19 +120,17 @@ int getPatch() { * @throws IllegalArgumentException if the value could not be parsed */ static LifecycleVersion parse(String value) { - Assert.hasText(value, "Value must not be empty"); - if (value.startsWith("v") || value.startsWith("V")) { - value = value.substring(1); - } - String[] components = value.split("\\."); - Assert.isTrue(components.length <= 3, "Malformed version number '" + value + "'"); + Assert.hasText(value, "'value' must not be empty"); + String withoutPrefix = (value.startsWith("v") || value.startsWith("V")) ? value.substring(1) : value; + String[] components = withoutPrefix.split("\\."); + Assert.isTrue(components.length <= 3, () -> "'value' [%s] must be a valid version number".formatted(value)); int[] versions = new int[3]; for (int i = 0; i < components.length; i++) { try { versions[i] = Integer.parseInt(components[i]); } catch (NumberFormatException ex) { - throw new IllegalArgumentException("Malformed version number '" + value + "'", ex); + throw new IllegalArgumentException("'value' [" + value + "] must be a valid version number", ex); } } return new LifecycleVersion(versions[0], versions[1], versions[2]); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/StackId.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/StackId.java index 327ca42fadac..6ed3903e7fb7 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/StackId.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/build/StackId.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -67,7 +67,7 @@ public String toString() { * @return the extracted stack ID */ static StackId fromImage(Image image) { - Assert.notNull(image, "Image must not be null"); + Assert.notNull(image, "'image' must not be null"); return fromImageConfig(image.getConfig()); } @@ -87,7 +87,7 @@ private static StackId fromImageConfig(ImageConfig imageConfig) { * @return a new stack ID instance */ static StackId of(String value) { - Assert.hasText(value, "Value must not be empty"); + Assert.hasText(value, "'value' must not be empty"); return new StackId(value); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/DockerApi.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/DockerApi.java index a61564b68c79..d3344654b0e3 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/DockerApi.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/DockerApi.java @@ -17,11 +17,8 @@ package org.springframework.boot.buildpack.platform.docker; import java.io.IOException; -import java.io.OutputStream; import java.net.URI; import java.net.URISyntaxException; -import java.nio.file.Files; -import java.nio.file.Path; import java.util.Arrays; import java.util.Collection; import java.util.Collections; @@ -90,7 +87,7 @@ public class DockerApi { * Create a new {@link DockerApi} instance. */ public DockerApi() { - this(HttpTransport.create(null)); + this(null); } /** @@ -99,21 +96,34 @@ public DockerApi() { * @since 2.4.0 */ public DockerApi(DockerHostConfiguration dockerHost) { - this(HttpTransport.create(dockerHost)); + this(dockerHost, DockerLog.toSystemOut()); + } + + /** + * Create a new {@link DockerApi} instance. + * @param dockerHost the Docker daemon host information + * @param log a logger used to record output + * @since 3.5.0 + */ + public DockerApi(DockerHostConfiguration dockerHost, DockerLog log) { + this(HttpTransport.create(dockerHost), log); } /** * Create a new {@link DockerApi} instance backed by a specific {@link HttpTransport} * implementation. * @param http the http implementation + * @param log a logger used to record output */ - DockerApi(HttpTransport http) { + DockerApi(HttpTransport http, DockerLog log) { + Assert.notNull(http, "'http' must not be null"); + Assert.notNull(log, "'log' must not be null"); this.http = http; this.jsonStream = new JsonStream(SharedObjectMapper.get()); this.image = new ImageApi(); this.container = new ContainerApi(); this.volume = new VolumeApi(); - this.system = new SystemApi(); + this.system = new SystemApi(log); } private HttpTransport http() { @@ -221,8 +231,8 @@ public Image pull(ImageReference reference, ImagePlatform platform, */ public Image pull(ImageReference reference, ImagePlatform platform, UpdateListener listener, String registryAuth) throws IOException { - Assert.notNull(reference, "Reference must not be null"); - Assert.notNull(listener, "Listener must not be null"); + Assert.notNull(reference, "'reference' must not be null"); + Assert.notNull(listener, "'listener' must not be null"); URI createUri = (platform != null) ? buildUrl(PLATFORM_API_VERSION, "/images/create", "fromImage", reference, "platform", platform) : buildUrl("/images/create", "fromImage", reference); @@ -251,8 +261,8 @@ public Image pull(ImageReference reference, ImagePlatform platform, */ public void push(ImageReference reference, UpdateListener listener, String registryAuth) throws IOException { - Assert.notNull(reference, "Reference must not be null"); - Assert.notNull(listener, "Listener must not be null"); + Assert.notNull(reference, "'reference' must not be null"); + Assert.notNull(listener, "'listener' must not be null"); URI pushUri = buildUrl("/images/" + reference + "/push"); ErrorCaptureUpdateListener errorListener = new ErrorCaptureUpdateListener(); listener.onStart(); @@ -276,8 +286,8 @@ public void push(ImageReference reference, UpdateListener * @throws IOException on IO error */ public void load(ImageArchive archive, UpdateListener listener) throws IOException { - Assert.notNull(archive, "Archive must not be null"); - Assert.notNull(listener, "Listener must not be null"); + Assert.notNull(archive, "'archive' must not be null"); + Assert.notNull(listener, "'listener' must not be null"); URI loadUri = buildUrl("/images/load"); LoadImageUpdateListener streamListener = new LoadImageUpdateListener(archive); listener.onStart(); @@ -295,34 +305,6 @@ public void load(ImageArchive archive, UpdateListener list } } - /** - * Export the layers of an image as paths to layer tar files. - * @param reference the reference to export - * @param exports a consumer to receive the layer tar file paths (file can only be - * accessed during the callback) - * @throws IOException on IO error - * @since 2.7.10 - * @deprecated since 3.2.6 for removal in 3.5.0 in favor of - * {@link #exportLayers(ImageReference, IOBiConsumer)} - */ - @Deprecated(since = "3.2.6", forRemoval = true) - public void exportLayerFiles(ImageReference reference, IOBiConsumer exports) throws IOException { - Assert.notNull(reference, "Reference must not be null"); - Assert.notNull(exports, "Exports must not be null"); - exportLayers(reference, (name, archive) -> { - Path path = Files.createTempFile("docker-export-layer-files-", null); - try { - try (OutputStream out = Files.newOutputStream(path)) { - archive.writeTo(out); - exports.accept(name, path); - } - } - finally { - Files.delete(path); - } - }); - } - /** * Export the layers of an image as {@link TarArchive TarArchives}. * @param reference the reference to export @@ -332,8 +314,8 @@ public void exportLayerFiles(ImageReference reference, IOBiConsumer exports) throws IOException { - Assert.notNull(reference, "Reference must not be null"); - Assert.notNull(exports, "Exports must not be null"); + Assert.notNull(reference, "'reference' must not be null"); + Assert.notNull(exports, "'exports' must not be null"); URI uri = buildUrl("/images/" + reference + "/get"); try (Response response = http().get(uri)) { try (ExportedImageTar exportedImageTar = new ExportedImageTar(reference, response.getContent())) { @@ -349,7 +331,7 @@ public void exportLayers(ImageReference reference, IOBiConsumer params = force ? FORCE_PARAMS : Collections.emptySet(); URI uri = buildUrl("/images/" + reference, params); http().delete(uri).close(); @@ -366,7 +348,7 @@ public Image inspect(ImageReference reference) throws IOException { } private Image inspect(ApiVersion apiVersion, ImageReference reference) throws IOException { - Assert.notNull(reference, "Reference must not be null"); + Assert.notNull(reference, "'reference' must not be null"); URI imageUri = buildUrl(apiVersion, "/images/" + reference + "/json"); try (Response response = http().get(imageUri)) { return Image.of(response.getContent()); @@ -374,8 +356,8 @@ private Image inspect(ApiVersion apiVersion, ImageReference reference) throws IO } public void tag(ImageReference sourceReference, ImageReference targetReference) throws IOException { - Assert.notNull(sourceReference, "SourceReference must not be null"); - Assert.notNull(targetReference, "TargetReference must not be null"); + Assert.notNull(sourceReference, "'sourceReference' must not be null"); + Assert.notNull(targetReference, "'targetReference' must not be null"); String tag = targetReference.getTag(); String path = "/images/" + sourceReference + "/tag"; URI uri = (tag != null) ? buildUrl(path, "repo", targetReference.inTaglessForm(), "tag", tag) @@ -404,8 +386,8 @@ public class ContainerApi { */ public ContainerReference create(ContainerConfig config, ImagePlatform platform, ContainerContent... contents) throws IOException { - Assert.notNull(config, "Config must not be null"); - Assert.noNullElements(contents, "Contents must not contain null elements"); + Assert.notNull(config, "'config' must not be null"); + Assert.noNullElements(contents, "'contents' must not contain null elements"); ContainerReference containerReference = createContainer(config, platform); for (ContainerContent content : contents) { uploadContainerContent(containerReference, content); @@ -434,7 +416,7 @@ private void uploadContainerContent(ContainerReference reference, ContainerConte * @throws IOException on IO error */ public void start(ContainerReference reference) throws IOException { - Assert.notNull(reference, "Reference must not be null"); + Assert.notNull(reference, "'reference' must not be null"); URI uri = buildUrl("/containers/" + reference + "/start"); http().post(uri).close(); } @@ -446,8 +428,8 @@ public void start(ContainerReference reference) throws IOException { * @throws IOException on IO error */ public void logs(ContainerReference reference, UpdateListener listener) throws IOException { - Assert.notNull(reference, "Reference must not be null"); - Assert.notNull(listener, "Listener must not be null"); + Assert.notNull(reference, "'reference' must not be null"); + Assert.notNull(listener, "'listener' must not be null"); Object[] params = { "stdout", "1", "stderr", "1", "follow", "1" }; URI uri = buildUrl("/containers/" + reference + "/logs", params); listener.onStart(); @@ -468,7 +450,7 @@ public void logs(ContainerReference reference, UpdateListener li * @throws IOException on IO error */ public ContainerStatus wait(ContainerReference reference) throws IOException { - Assert.notNull(reference, "Reference must not be null"); + Assert.notNull(reference, "'reference' must not be null"); URI uri = buildUrl("/containers/" + reference + "/wait"); try (Response response = http().post(uri)) { return ContainerStatus.of(response.getContent()); @@ -482,7 +464,7 @@ public ContainerStatus wait(ContainerReference reference) throws IOException { * @throws IOException on IO error */ public void remove(ContainerReference reference, boolean force) throws IOException { - Assert.notNull(reference, "Reference must not be null"); + Assert.notNull(reference, "'reference' must not be null"); Collection params = force ? FORCE_PARAMS : Collections.emptySet(); URI uri = buildUrl("/containers/" + reference, params); http().delete(uri).close(); @@ -505,7 +487,7 @@ public class VolumeApi { * @throws IOException on IO error */ public void delete(VolumeName name, boolean force) throws IOException { - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); Collection params = force ? FORCE_PARAMS : Collections.emptySet(); URI uri = buildUrl("/volumes/" + name, params); http().delete(uri).close(); @@ -518,7 +500,10 @@ public void delete(VolumeName name, boolean force) throws IOException { */ class SystemApi { - SystemApi() { + private final DockerLog log; + + SystemApi(DockerLog log) { + this.log = log; } /** @@ -535,6 +520,7 @@ ApiVersion getApiVersion() { } } catch (Exception ex) { + this.log.log("Warning: Failed to determine Docker API version: " + ex.getMessage()); // fall through to return default value } return UNKNOWN_API_VERSION; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/DockerLog.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/DockerLog.java new file mode 100644 index 000000000000..a0cad1660326 --- /dev/null +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/DockerLog.java @@ -0,0 +1,54 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.buildpack.platform.docker; + +import java.io.PrintStream; + +/** + * Callback interface used to provide {@link DockerApi} output logging. + * + * @author Dmytro Nosan + * @since 3.5.0 + * @see #toSystemOut() + */ +public interface DockerLog { + + /** + * Logs a given message. + * @param message the message to log + */ + void log(String message); + + /** + * Factory method that returns a {@link DockerLog} that outputs to {@link System#out}. + * @return {@link DockerLog} instance that logs to system out + */ + static DockerLog toSystemOut() { + return to(System.out); + } + + /** + * Factory method that returns a {@link DockerLog} that outputs to a given + * {@link PrintStream}. + * @param out the print stream used to output the log + * @return {@link DockerLog} instance that logs to the given print stream + */ + static DockerLog to(PrintStream out) { + return out::println; + } + +} diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ExportedImageTar.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ExportedImageTar.java index f04cfef7fcdd..1e20e3611e7b 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ExportedImageTar.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ExportedImageTar.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -127,7 +127,8 @@ static LayerArchiveFactory create(ImageReference reference, Path tarFile) throws entry = tar.getNextEntry(); } Assert.state(index != null || manifest != null, - "Exported image '%s' does not contain 'index.json' or 'manifest.json'".formatted(reference)); + () -> "Exported image '%s' does not contain 'index.json' or 'manifest.json'" + .formatted(reference)); return (index != null) ? new IndexLayerArchiveFactory(tarFile, index) : new ManifestLayerArchiveFactory(tarFile, manifest); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ProgressUpdateEvent.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ProgressUpdateEvent.java index 6ac289436341..986004d1ddc3 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ProgressUpdateEvent.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ProgressUpdateEvent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -78,28 +78,6 @@ public ProgressDetail(Long current, Long total) { this.total = total; } - /** - * Return the current progress value. - * @return the current progress - * @deprecated since 3.3.7 for removal in 3.5.0 in favor of - * {@link #asPercentage()} - */ - @Deprecated(since = "3.3.7", forRemoval = true) - public int getCurrent() { - return (int) Long.min(this.current, Integer.MAX_VALUE); - } - - /** - * Return the total progress possible value. - * @return the total progress possible - * @deprecated since 3.3.7 for removal in 3.5.0 in favor of - * {@link #asPercentage()} - */ - @Deprecated(since = "3.3.7", forRemoval = true) - public int getTotal() { - return (int) Long.min(this.total, Integer.MAX_VALUE); - } - /** * Return the progress as a percentage. * @return the progress percentage @@ -110,14 +88,7 @@ public int asPercentage() { return (percentage < 0) ? 0 : Math.min(percentage, 100); } - /** - * Return if the progress detail is considered empty. - * @param progressDetail the progress detail to check - * @return if the progress detail is empty - * @deprecated since 3.3.7 for removal in 3.5.0 - */ - @Deprecated(since = "3.3.7", forRemoval = true) - public static boolean isEmpty(ProgressDetail progressDetail) { + private static boolean isEmpty(ProgressDetail progressDetail) { return progressDetail == null || progressDetail.current == null || progressDetail.total == null; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/TotalProgressEvent.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/TotalProgressEvent.java index e14b5d661db2..4e624d5d9ec3 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/TotalProgressEvent.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/TotalProgressEvent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ public class TotalProgressEvent { * @param percent the progress as a percentage */ public TotalProgressEvent(int percent) { - Assert.isTrue(percent >= 0 && percent <= 100, "Percent must be in the range 0 to 100"); + Assert.isTrue(percent >= 0 && percent <= 100, "'percent' must be in the range 0 to 100"); this.percent = percent; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/TotalProgressListener.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/TotalProgressListener.java index 9c5e3395f628..91db615d37cb 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/TotalProgressListener.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/TotalProgressListener.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/configuration/DockerConfiguration.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/configuration/DockerConfiguration.java index fa47c349fbd1..e04f4c9cb10e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/configuration/DockerConfiguration.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/configuration/DockerConfiguration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -64,13 +64,13 @@ public DockerRegistryAuthentication getPublishRegistryAuthentication() { } public DockerConfiguration withHost(String address, boolean secure, String certificatePath) { - Assert.notNull(address, "Address must not be null"); + Assert.notNull(address, "'address' must not be null"); return new DockerConfiguration(DockerHostConfiguration.forAddress(address, secure, certificatePath), this.builderAuthentication, this.publishAuthentication, this.bindHostToBuilder); } public DockerConfiguration withContext(String context) { - Assert.notNull(context, "Context must not be null"); + Assert.notNull(context, "'context' must not be null"); return new DockerConfiguration(DockerHostConfiguration.forContext(context), this.builderAuthentication, this.publishAuthentication, this.bindHostToBuilder); } @@ -81,29 +81,29 @@ public DockerConfiguration withBindHostToBuilder(boolean bindHostToBuilder) { } public DockerConfiguration withBuilderRegistryTokenAuthentication(String token) { - Assert.notNull(token, "Token must not be null"); + Assert.notNull(token, "'token' must not be null"); return new DockerConfiguration(this.host, new DockerRegistryTokenAuthentication(token), this.publishAuthentication, this.bindHostToBuilder); } public DockerConfiguration withBuilderRegistryUserAuthentication(String username, String password, String url, String email) { - Assert.notNull(username, "Username must not be null"); - Assert.notNull(password, "Password must not be null"); + Assert.notNull(username, "'username' must not be null"); + Assert.notNull(password, "'password' must not be null"); return new DockerConfiguration(this.host, new DockerRegistryUserAuthentication(username, password, url, email), this.publishAuthentication, this.bindHostToBuilder); } public DockerConfiguration withPublishRegistryTokenAuthentication(String token) { - Assert.notNull(token, "Token must not be null"); + Assert.notNull(token, "'token' must not be null"); return new DockerConfiguration(this.host, this.builderAuthentication, new DockerRegistryTokenAuthentication(token), this.bindHostToBuilder); } public DockerConfiguration withPublishRegistryUserAuthentication(String username, String password, String url, String email) { - Assert.notNull(username, "Username must not be null"); - Assert.notNull(password, "Password must not be null"); + Assert.notNull(username, "'username' must not be null"); + Assert.notNull(password, "'password' must not be null"); return new DockerConfiguration(this.host, this.builderAuthentication, new DockerRegistryUserAuthentication(username, password, url, email), this.bindHostToBuilder); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ssl/PemPrivateKeyParser.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ssl/PemPrivateKeyParser.java index 45e4e5c9b473..9f1275eea3b3 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ssl/PemPrivateKeyParser.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/ssl/PemPrivateKeyParser.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -445,7 +445,7 @@ static class Pkcs8PrivateKeyDecryptor { public static final String PBES2_ALGORITHM = "PBES2"; static PKCS8EncodedKeySpec decrypt(byte[] bytes, String password) { - Assert.notNull(password, "Password is required for an encrypted private key"); + Assert.state(password != null, "Password is required for an encrypted private key"); try { EncryptedPrivateKeyInfo keyInfo = new EncryptedPrivateKeyInfo(bytes); AlgorithmParameters algorithmParameters = keyInfo.getAlgParameters(); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/DockerConnectionException.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/DockerConnectionException.java index ec22850faa17..955c5fde151b 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/DockerConnectionException.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/DockerConnectionException.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,8 +34,8 @@ public DockerConnectionException(String host, Exception cause) { } private static String buildMessage(String host, Exception cause) { - Assert.notNull(host, "Host must not be null"); - Assert.notNull(cause, "Cause must not be null"); + Assert.notNull(host, "'host' must not be null"); + Assert.notNull(cause, "'cause' must not be null"); StringBuilder message = new StringBuilder("Connection to the Docker daemon at '" + host + "' failed"); String causeMessage = getCauseMessage(cause); if (StringUtils.hasText(causeMessage)) { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/DockerEngineException.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/DockerEngineException.java index e521f4a8afa4..d25698186863 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/DockerEngineException.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/DockerEngineException.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -83,8 +83,8 @@ public Message getResponseMessage() { private static String buildMessage(String host, URI uri, int statusCode, String reasonPhrase, Errors errors, Message responseMessage) { - Assert.notNull(host, "Host must not be null"); - Assert.notNull(uri, "URI must not be null"); + Assert.notNull(host, "'host' must not be null"); + Assert.notNull(uri, "'uri' must not be null"); StringBuilder message = new StringBuilder( "Docker API call to '" + host + uri + "' failed with status code " + statusCode); if (StringUtils.hasLength(reasonPhrase)) { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/HttpClientTransport.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/HttpClientTransport.java index 28eaf45715b9..3546ab834f2f 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/HttpClientTransport.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/HttpClientTransport.java @@ -61,8 +61,8 @@ abstract class HttpClientTransport implements HttpTransport { private final HttpHost host; protected HttpClientTransport(HttpClient client, HttpHost host) { - Assert.notNull(client, "Client must not be null"); - Assert.notNull(host, "Host must not be null"); + Assert.notNull(client, "'client' must not be null"); + Assert.notNull(host, "'host' must not be null"); this.client = client; this.host = host; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/RemoteHttpClientTransport.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/RemoteHttpClientTransport.java index 1b8d84260d85..6eed8043bb7e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/RemoteHttpClientTransport.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/transport/RemoteHttpClientTransport.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,6 +35,7 @@ import org.springframework.boot.buildpack.platform.docker.configuration.ResolvedDockerHost; import org.springframework.boot.buildpack.platform.docker.ssl.SslContextFactory; import org.springframework.util.Assert; +import org.springframework.util.StringUtils; /** * {@link HttpClientTransport} that talks to a remote Docker. @@ -85,8 +86,8 @@ private static RemoteHttpClientTransport create(DockerHost host, SslContextFacto private static TlsSocketStrategy getTlsSocketStrategy(DockerHost host, SslContextFactory sslContextFactory) { String directory = host.getCertificatePath(); - Assert.hasText(directory, - () -> "Docker host TLS verification requires trust material location to be specified with certificate path"); + Assert.state(StringUtils.hasText(directory), + "Docker host TLS verification requires trust material location to be specified with certificate path"); SSLContext sslContext = sslContextFactory.forDirectory(directory); return new DefaultClientTlsStrategy(sslContext); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ApiVersion.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ApiVersion.java index 662680f4bd42..b7316bc62a0f 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ApiVersion.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ApiVersion.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -118,16 +118,17 @@ public String toString() { * @throws IllegalArgumentException if the value could not be parsed */ public static ApiVersion parse(String value) { - Assert.hasText(value, "Value must not be empty"); + Assert.hasText(value, "'value' must not be empty"); Matcher matcher = PATTERN.matcher(value); - Assert.isTrue(matcher.matches(), () -> "Malformed version number '" + value + "'"); + Assert.isTrue(matcher.matches(), + () -> "'value' [%s] must contain a well formed version number".formatted(value)); try { int major = Integer.parseInt(matcher.group(1)); int minor = Integer.parseInt(matcher.group(2)); return new ApiVersion(major, minor); } catch (NumberFormatException ex) { - throw new IllegalArgumentException("Malformed version number '" + value + "'", ex); + throw new IllegalArgumentException("'value' must contain a well formed version number [" + value + "]", ex); } } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/Binding.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/Binding.java index 7d08240af857..caa3f1501e71 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/Binding.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/Binding.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -110,7 +110,7 @@ private List getParts() { * @return a new {@link Binding} instance */ public static Binding of(String value) { - Assert.notNull(value, "Value must not be null"); + Assert.notNull(value, "'value' must not be null"); return new Binding(value); } @@ -121,7 +121,7 @@ public static Binding of(String value) { * @return a new {@link Binding} instance */ public static Binding from(VolumeName sourceVolume, String destination) { - Assert.notNull(sourceVolume, "SourceVolume must not be null"); + Assert.notNull(sourceVolume, "'sourceVolume' must not be null"); return from(sourceVolume.toString(), destination); } @@ -132,8 +132,8 @@ public static Binding from(VolumeName sourceVolume, String destination) { * @return a new {@link Binding} instance */ public static Binding from(String source, String destination) { - Assert.notNull(source, "Source must not be null"); - Assert.notNull(destination, "Destination must not be null"); + Assert.notNull(source, "'source' must not be null"); + Assert.notNull(destination, "'destination' must not be null"); return new Binding(source + ":" + destination); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerConfig.java index 1ee0e19ae6f3..c13e737ad819 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -51,8 +51,8 @@ public class ContainerConfig { ContainerConfig(String user, ImageReference image, String command, List args, Map labels, List bindings, Map env, String networkMode, List securityOptions) throws IOException { - Assert.notNull(image, "Image must not be null"); - Assert.hasText(command, "Command must not be empty"); + Assert.notNull(image, "'image' must not be null"); + Assert.hasText(command, "'command' must not be empty"); ObjectMapper objectMapper = SharedObjectMapper.get(); ObjectNode node = objectMapper.createObjectNode(); if (StringUtils.hasText(user)) { @@ -100,8 +100,8 @@ public String toString() { * @return a new {@link ContainerConfig} instance */ public static ContainerConfig of(ImageReference imageReference, Consumer update) { - Assert.notNull(imageReference, "ImageReference must not be null"); - Assert.notNull(update, "Update must not be null"); + Assert.notNull(imageReference, "'imageReference' must not be null"); + Assert.notNull(update, "'update' must not be null"); return new Update(imageReference).run(update); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerContent.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerContent.java index c0f5dd217d49..5a9b04b20767 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerContent.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerContent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,8 +56,8 @@ static ContainerContent of(TarArchive archive) { * @return a new {@link ContainerContent} instance */ static ContainerContent of(TarArchive archive, String destinationPath) { - Assert.notNull(archive, "Archive must not be null"); - Assert.hasText(destinationPath, "DestinationPath must not be empty"); + Assert.notNull(archive, "'archive' must not be null"); + Assert.hasText(destinationPath, "'destinationPath' must not be empty"); return new ContainerContent() { @Override diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerReference.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerReference.java index f5a5c6ac5a39..a26d43feaadd 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerReference.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ContainerReference.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ public final class ContainerReference { private final String value; private ContainerReference(String value) { - Assert.hasText(value, "Value must not be empty"); + Assert.hasText(value, "'value' must not be empty"); this.value = value; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageArchive.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageArchive.java index 8d8145ac4113..81b4565f72de 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageArchive.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageArchive.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -289,7 +289,7 @@ public void withUpdatedConfig(Consumer update) { * @param layer the layer to add */ public void withNewLayer(Layer layer) { - Assert.notNull(layer, "Layer must not be null"); + Assert.notNull(layer, "'layer' must not be null"); this.newLayers.add(layer); } @@ -298,7 +298,7 @@ public void withNewLayer(Layer layer) { * @param createDate the create date */ public void withCreateDate(Instant createDate) { - Assert.notNull(createDate, "CreateDate must not be null"); + Assert.notNull(createDate, "'createDate' must not be null"); this.createDate = createDate; } @@ -307,7 +307,7 @@ public void withCreateDate(Instant createDate) { * @param tag the tag */ public void withTag(ImageReference tag) { - Assert.notNull(tag, "Tag must not be null"); + Assert.notNull(tag, "'tag' must not be null"); this.tag = tag.inTaggedForm(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageName.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageName.java index 073871c39692..fa9a85c4ad72 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageName.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageName.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,7 +42,7 @@ public class ImageName { private final String string; ImageName(String domain, String path) { - Assert.hasText(path, "Path must not be empty"); + Assert.hasText(path, "'path' must not be empty"); this.domain = getDomainOrDefault(domain); this.name = getNameWithDefaultPath(this.domain, path); this.string = this.domain + "/" + this.name; @@ -126,13 +126,12 @@ private String getNameWithDefaultPath(String domain, String name) { * @return an {@link ImageName} instance */ public static ImageName of(String value) { - Assert.hasText(value, "Value must not be empty"); + Assert.hasText(value, "'value' must not be empty"); String domain = parseDomain(value); String path = (domain != null) ? value.substring(domain.length() + 1) : value; Assert.isTrue(Regex.PATH.matcher(path).matches(), - () -> "Unable to parse name \"" + value + "\". " - + "Image name must be in the form '[domainHost:port/][path/]name', " - + "with 'path' and 'name' containing only [a-z0-9][.][_][-]"); + () -> "'value' [" + value + "] must be a parsable name in the form '[domainHost:port/][path/]name' (" + + "with 'path' and 'name' containing only [a-z0-9][.][_][-])"); return new ImageName(domain, path); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImagePlatform.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImagePlatform.java index d43570c117f9..4b6ce2bc8928 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImagePlatform.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImagePlatform.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,7 +35,7 @@ public class ImagePlatform { private final String variant; ImagePlatform(String os, String architecture, String variant) { - Assert.hasText(os, "OS must not be empty"); + Assert.hasText(os, "'os' must not be empty"); this.os = os; this.architecture = architecture; this.variant = variant; @@ -78,14 +78,14 @@ public String toString() { * @return an {@link ImagePlatform} instance */ public static ImagePlatform of(String value) { - Assert.hasText(value, "Value must not be empty"); + Assert.hasText(value, "'value' must not be empty"); String[] split = value.split("/+"); return switch (split.length) { case 1 -> new ImagePlatform(split[0], null, null); case 2 -> new ImagePlatform(split[0], split[1], null); case 3 -> new ImagePlatform(split[0], split[1], split[2]); default -> throw new IllegalArgumentException( - "ImagePlatform value '" + value + "' must be in the form of os[/architecture[/variant]]"); + "'value' [" + value + "] must be in the form 'os[/architecture[/variant]]'"); }; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageReference.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageReference.java index c3c263e89bf8..7bf34f67bd9d 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageReference.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/ImageReference.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,7 +48,7 @@ public final class ImageReference { private final String string; private ImageReference(ImageName name, String tag, String digest) { - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); this.name = name; this.tag = tag; this.digest = digest; @@ -186,8 +186,10 @@ public ImageReference inTaggedOrDigestForm() { * @return an {@link ImageName} for the jar file. */ public static ImageReference forJarFile(File jarFile) { + Assert.notNull(jarFile, "'jarFile' must not be null"); String filename = jarFile.getName(); - Assert.isTrue(filename.toLowerCase(Locale.ROOT).endsWith(".jar"), () -> "File '" + jarFile + "' is not a JAR"); + Assert.isTrue(filename.toLowerCase(Locale.ROOT).endsWith(".jar"), + () -> "'jarFile' must end with '.jar' [" + jarFile + "]"); filename = filename.substring(0, filename.length() - 4); int firstDot = filename.indexOf('.'); if (firstDot == -1) { @@ -236,7 +238,7 @@ public static ImageReference random(String prefix, int randomLength) { * @return an {@link ImageName} instance */ public static ImageReference of(String value) { - Assert.hasText(value, "Value must not be null"); + Assert.hasText(value, "'value' must not be null"); String domain = ImageName.parseDomain(value); String path = (domain != null) ? value.substring(domain.length() + 1) : value; String digest = null; @@ -261,11 +263,10 @@ public static ImageReference of(String value) { path = path.substring(0, tagSplit) + remainder; } } - Assert.isTrue(isLowerCase(path) && matchesPathRegex(path), - () -> "Unable to parse image reference \"" + value + "\". " - + "Image reference must be in the form '[domainHost:port/][path/]name[:tag][@digest]', " - + "with 'path' and 'name' containing only [a-z0-9][.][_][-]"); + () -> "'value' [" + value + "] must be an image reference in the form " + + "'[domainHost:port/][path/]name[:tag][@digest]' " + + "(with 'path' and 'name' containing only [a-z0-9][.][_][-])"); ImageName name = new ImageName(domain, path); return new ImageReference(name, tag, digest); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/Layer.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/Layer.java index 6d481eb12d7f..78671a1faaa6 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/Layer.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/Layer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -71,7 +71,7 @@ public void writeTo(OutputStream outputStream) throws IOException { * @throws IOException on IO error */ public static Layer of(IOConsumer layout) throws IOException { - Assert.notNull(layout, "Layout must not be null"); + Assert.notNull(layout, "'layout' must not be null"); return fromTarArchive(TarArchive.of(layout)); } @@ -82,7 +82,7 @@ public static Layer of(IOConsumer layout) throws IOException { * @throws IOException on error */ public static Layer fromTarArchive(TarArchive tarArchive) throws IOException { - Assert.notNull(tarArchive, "TarArchive must not be null"); + Assert.notNull(tarArchive, "'tarArchive' must not be null"); try { return new Layer(tarArchive); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/LayerId.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/LayerId.java index d30cc6a331aa..adc0d27f2856 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/LayerId.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/LayerId.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -83,9 +83,9 @@ public String toString() { * @return a new layer ID instance */ public static LayerId of(String value) { - Assert.hasText(value, "Value must not be empty"); + Assert.hasText(value, "'value' must not be empty"); int i = value.indexOf(':'); - Assert.isTrue(i >= 0, () -> "Invalid layer ID '" + value + "'"); + Assert.isTrue(i >= 0, () -> "'value' [%s] must contain a valid layer ID".formatted(value)); return new LayerId(value, value.substring(0, i), value.substring(i + 1)); } @@ -95,8 +95,8 @@ public static LayerId of(String value) { * @return a new layer ID instance */ public static LayerId ofSha256Digest(byte[] digest) { - Assert.notNull(digest, "Digest must not be null"); - Assert.isTrue(digest.length == 32, "Digest must be exactly 32 bytes"); + Assert.notNull(digest, "'digest' must not be null"); + Assert.isTrue(digest.length == 32, "'digest' must be exactly 32 bytes"); String algorithm = "sha256"; String hash = String.format("%064x", new BigInteger(1, digest)); return new LayerId(algorithm + ":" + hash, algorithm, hash); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/RandomString.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/RandomString.java index d9696d3db7f3..3cf51ae9fca1 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/RandomString.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/RandomString.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ private RandomString() { } static String generate(String prefix, int randomLength) { - Assert.notNull(prefix, "Prefix must not be null"); + Assert.notNull(prefix, "'prefix' must not be null"); return prefix + generateRandom(randomLength); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/VolumeName.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/VolumeName.java index 7ea8a18a1cb4..d3619ddd8ba2 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/VolumeName.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/docker/type/VolumeName.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -106,10 +106,10 @@ public static VolumeName basedOn(S source, String prefix, String suffix, int */ public static VolumeName basedOn(S source, Function nameExtractor, String prefix, String suffix, int digestLength) { - Assert.notNull(source, "Source must not be null"); - Assert.notNull(nameExtractor, "NameExtractor must not be null"); - Assert.notNull(prefix, "Prefix must not be null"); - Assert.notNull(suffix, "Suffix must not be null"); + Assert.notNull(source, "'source' must not be null"); + Assert.notNull(nameExtractor, "'nameExtractor' must not be null"); + Assert.notNull(prefix, "'prefix' must not be null"); + Assert.notNull(suffix, "'suffix' must not be null"); return of(prefix + getDigest(nameExtractor.apply(source), digestLength) + suffix); } @@ -125,7 +125,7 @@ private static String getDigest(String name, int length) { private static String asHexString(byte[] digest, int digestLength) { Assert.isTrue(digestLength <= digest.length, - () -> "DigestLength must be less than or equal to " + digest.length); + () -> "'digestLength' must be less than or equal to " + digest.length); return HexFormat.of().formatHex(digest, 0, digestLength); } @@ -135,7 +135,7 @@ private static String asHexString(byte[] digest, int digestLength) { * @return a new {@link VolumeName} instance */ public static VolumeName of(String value) { - Assert.notNull(value, "Value must not be null"); + Assert.notNull(value, "'value' must not be null"); return new VolumeName(value); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/Content.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/Content.java index 911df2ad4927..367dd9b4276f 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/Content.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/Content.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -54,7 +54,7 @@ public interface Content { * @return a new {@link Content} instance */ static Content of(String string) { - Assert.notNull(string, "String must not be null"); + Assert.notNull(string, "'string' must not be null"); return of(string.getBytes(StandardCharsets.UTF_8)); } @@ -64,7 +64,7 @@ static Content of(String string) { * @return a new {@link Content} instance */ static Content of(byte[] bytes) { - Assert.notNull(bytes, "Bytes must not be null"); + Assert.notNull(bytes, "'bytes' must not be null"); return of(bytes.length, () -> new ByteArrayInputStream(bytes)); } @@ -74,7 +74,7 @@ static Content of(byte[] bytes) { * @return a new {@link Content} instance */ static Content of(File file) { - Assert.notNull(file, "File must not be null"); + Assert.notNull(file, "'file' must not be null"); return of((int) file.length(), () -> new FileInputStream(file)); } @@ -86,8 +86,8 @@ static Content of(File file) { * @return a new {@link Content} instance */ static Content of(int size, IOSupplier supplier) { - Assert.isTrue(size >= 0, "Size must not be negative"); - Assert.notNull(supplier, "Supplier must not be null"); + Assert.isTrue(size >= 0, "'size' must not be negative"); + Assert.notNull(supplier, "'supplier' must not be null"); return new Content() { @Override diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/FilePermissions.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/FilePermissions.java index 7f63c012d82e..831de4819e24 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/FilePermissions.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/FilePermissions.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -45,7 +45,7 @@ private FilePermissions() { * @throws IOException if path permissions cannot be read */ public static int umaskForPath(Path path) throws IOException { - Assert.notNull(path, "Path must not be null"); + Assert.notNull(path, "'path' must not be null"); PosixFileAttributeView attributeView = Files.getFileAttributeView(path, PosixFileAttributeView.class); Assert.state(attributeView != null, "Unsupported file type for retrieving Posix attributes"); return posixPermissionsToUmask(attributeView.readAttributes().permissions()); @@ -59,7 +59,7 @@ public static int umaskForPath(Path path) throws IOException { * @return the integer representation */ public static int posixPermissionsToUmask(Collection permissions) { - Assert.notNull(permissions, "Permissions must not be null"); + Assert.notNull(permissions, "'permissions' must not be null"); int owner = permissionToUmask(permissions, PosixFilePermission.OWNER_EXECUTE, PosixFilePermission.OWNER_WRITE, PosixFilePermission.OWNER_READ); int group = permissionToUmask(permissions, PosixFilePermission.GROUP_EXECUTE, PosixFilePermission.GROUP_WRITE, diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/InspectedContent.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/InspectedContent.java index 1f42a68d3f0c..6b9b9d0e3703 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/InspectedContent.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/InspectedContent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -76,7 +76,7 @@ else if (this.content instanceof File file) { * @throws IOException on IO error */ public static InspectedContent of(InputStream inputStream, Inspector... inspectors) throws IOException { - Assert.notNull(inputStream, "InputStream must not be null"); + Assert.notNull(inputStream, "'inputStream' must not be null"); return of((outputStream) -> FileCopyUtils.copy(inputStream, outputStream), inspectors); } @@ -88,7 +88,7 @@ public static InspectedContent of(InputStream inputStream, Inspector... inspecto * @throws IOException on IO error */ public static InspectedContent of(Content content, Inspector... inspectors) throws IOException { - Assert.notNull(content, "Content must not be null"); + Assert.notNull(content, "'content' must not be null"); return of(content::writeTo, inspectors); } @@ -101,7 +101,7 @@ public static InspectedContent of(Content content, Inspector... inspectors) thro * @throws IOException on IO error */ public static InspectedContent of(IOConsumer writer, Inspector... inspectors) throws IOException { - Assert.notNull(writer, "Writer must not be null"); + Assert.notNull(writer, "'writer' must not be null"); InspectingOutputStream outputStream = new InspectingOutputStream(inspectors); try (outputStream) { writer.accept(outputStream); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/ZipFileTarArchive.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/ZipFileTarArchive.java index dcbb3229b927..c1d3b100211e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/ZipFileTarArchive.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/main/java/org/springframework/boot/buildpack/platform/io/ZipFileTarArchive.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -53,8 +53,8 @@ public class ZipFileTarArchive implements TarArchive { * @param owner the owner of the tar entries */ public ZipFileTarArchive(File zip, Owner owner) { - Assert.notNull(zip, "Zip must not be null"); - Assert.notNull(owner, "Owner must not be null"); + Assert.notNull(zip, "'zip' must not be null"); + Assert.notNull(owner, "'owner' must not be null"); assertArchiveHasEntries(zip); this.zip = zip; this.owner = owner; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildOwnerTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildOwnerTests.java index c16232cfa9d1..4d04bdb22f4a 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildOwnerTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildOwnerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,7 +47,7 @@ void fromEnvReturnsOwner() { @Test void fromEnvWhenEnvIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> BuildOwner.fromEnv(null)) - .withMessage("Env must not be null"); + .withMessage("'env' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildRequestTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildRequestTests.java index 2d5490f7285a..2bf5de0ea211 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildRequestTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildRequestTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -88,20 +88,20 @@ void forJarFileWithNameReturnsRequest() throws IOException { @Test void forJarFileWhenJarFileIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> BuildRequest.forJarFile(null)) - .withMessage("JarFile must not be null"); + .withMessage("'jarFile' must not be null"); } @Test void forJarFileWhenJarFileIsMissingThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> BuildRequest.forJarFile(new File(this.tempDir, "missing.jar"))) - .withMessage("JarFile must exist"); + .withMessage("'jarFile' must exist"); } @Test void forJarFileWhenJarFileIsDirectoryThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> BuildRequest.forJarFile(this.tempDir)) - .withMessage("JarFile must be a file"); + .withMessage("'jarFile' must be a file"); } @Test @@ -217,14 +217,14 @@ void withEnvMapAddsEnvEntries() throws IOException { void withEnvWhenKeyIsNullThrowsException() throws IOException { BuildRequest request = BuildRequest.forJarFile(writeTestJarFile("my-app-0.0.1.jar")); assertThatIllegalArgumentException().isThrownBy(() -> request.withEnv(null, "test")) - .withMessage("Name must not be empty"); + .withMessage("'name' must not be empty"); } @Test void withEnvWhenValueIsNullThrowsException() throws IOException { BuildRequest request = BuildRequest.forJarFile(writeTestJarFile("my-app-0.0.1.jar")); assertThatIllegalArgumentException().isThrownBy(() -> request.withEnv("test", null)) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test @@ -241,7 +241,7 @@ void withBuildpacksAddsBuildpacks() throws IOException { void withBuildpacksWhenBuildpacksIsNullThrowsException() throws IOException { BuildRequest request = BuildRequest.forJarFile(writeTestJarFile("my-app-0.0.1.jar")); assertThatIllegalArgumentException().isThrownBy(() -> request.withBuildpacks((List) null)) - .withMessage("Buildpacks must not be null"); + .withMessage("'buildpacks' must not be null"); } @Test @@ -258,7 +258,7 @@ void withBindingsAddsBindings() throws IOException { void withBindingsWhenBindingsIsNullThrowsException() throws IOException { BuildRequest request = BuildRequest.forJarFile(writeTestJarFile("my-app-0.0.1.jar")); assertThatIllegalArgumentException().isThrownBy(() -> request.withBindings((List) null)) - .withMessage("Bindings must not be null"); + .withMessage("'bindings' must not be null"); } @Test @@ -283,7 +283,7 @@ void withTagsAddsTags() throws IOException { void withTagsWhenTagsIsNullThrowsException() throws IOException { BuildRequest request = BuildRequest.forJarFile(writeTestJarFile("my-app-0.0.1.jar")); assertThatIllegalArgumentException().isThrownBy(() -> request.withTags((List) null)) - .withMessage("Tags must not be null"); + .withMessage("'tags' must not be null"); } @Test @@ -322,7 +322,7 @@ void withBuildBindCacheAddsCache() throws IOException { void withBuildVolumeCacheWhenCacheIsNullThrowsException() throws IOException { BuildRequest request = BuildRequest.forJarFile(writeTestJarFile("my-app-0.0.1.jar")); assertThatIllegalArgumentException().isThrownBy(() -> request.withBuildCache(null)) - .withMessage("BuildCache must not be null"); + .withMessage("'buildCache' must not be null"); } @Test @@ -345,7 +345,7 @@ void withLaunchBindCacheAddsCache() throws IOException { void withLaunchVolumeCacheWhenCacheIsNullThrowsException() throws IOException { BuildRequest request = BuildRequest.forJarFile(writeTestJarFile("my-app-0.0.1.jar")); assertThatIllegalArgumentException().isThrownBy(() -> request.withLaunchCache(null)) - .withMessage("LaunchCache must not be null"); + .withMessage("'launchCache' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderBuildpackTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderBuildpackTests.java index d19666afdbf6..d3966a0ccb40 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderBuildpackTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderBuildpackTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ import org.springframework.boot.buildpack.platform.json.AbstractJsonTests; import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; import static org.mockito.BDDMockito.given; import static org.mockito.Mockito.mock; @@ -86,7 +86,7 @@ void resolveWhenUnqualifiedBuildpackWithoutVersionResolves() throws Exception { @Test void resolveWhenFullyQualifiedBuildpackWithVersionNotInBuilderThrowsException() { BuildpackReference reference = BuildpackReference.of("urn:cnb:builder:example/buildpack1@1.2.3"); - assertThatIllegalArgumentException().isThrownBy(() -> BuilderBuildpack.resolve(this.resolverContext, reference)) + assertThatIllegalStateException().isThrownBy(() -> BuilderBuildpack.resolve(this.resolverContext, reference)) .withMessageContaining("'urn:cnb:builder:example/buildpack1@1.2.3'") .withMessageContaining("not found in builder"); } @@ -94,7 +94,7 @@ void resolveWhenFullyQualifiedBuildpackWithVersionNotInBuilderThrowsException() @Test void resolveWhenFullyQualifiedBuildpackWithoutVersionNotInBuilderThrowsException() { BuildpackReference reference = BuildpackReference.of("urn:cnb:builder:example/buildpack1"); - assertThatIllegalArgumentException().isThrownBy(() -> BuilderBuildpack.resolve(this.resolverContext, reference)) + assertThatIllegalStateException().isThrownBy(() -> BuilderBuildpack.resolve(this.resolverContext, reference)) .withMessageContaining("'urn:cnb:builder:example/buildpack1'") .withMessageContaining("not found in builder"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderMetadataTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderMetadataTests.java index fd6803401d7b..5e760f873611 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderMetadataTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderMetadataTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,6 +28,7 @@ import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; import static org.assertj.core.api.Assertions.tuple; import static org.mockito.BDDMockito.given; import static org.mockito.Mockito.mock; @@ -89,14 +90,14 @@ void fromImageWithoutStackLoadsMetadata() throws IOException { @Test void fromImageWhenImageIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> BuilderMetadata.fromImage(null)) - .withMessage("Image must not be null"); + .withMessage("'image' must not be null"); } @Test void fromImageWhenImageConfigIsNullThrowsException() { Image image = mock(Image.class); assertThatIllegalArgumentException().isThrownBy(() -> BuilderMetadata.fromImage(image)) - .withMessage("ImageConfig must not be null"); + .withMessage("'imageConfig' must not be null"); } @Test @@ -105,7 +106,7 @@ void fromImageConfigWhenLabelIsMissingThrowsException() { ImageConfig imageConfig = mock(ImageConfig.class); given(image.getConfig()).willReturn(imageConfig); given(imageConfig.getLabels()).willReturn(Collections.singletonMap("alpha", "a")); - assertThatIllegalArgumentException().isThrownBy(() -> BuilderMetadata.fromImage(image)) + assertThatIllegalStateException().isThrownBy(() -> BuilderMetadata.fromImage(image)) .withMessage("No 'io.buildpacks.builder.metadata' label found in image config labels 'alpha'"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderTests.java index 239143aa477e..7cba312d72e5 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuilderTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,10 +25,12 @@ import org.mockito.ArgumentCaptor; import org.mockito.stubbing.Answer; +import org.springframework.boot.buildpack.platform.build.Builder.BuildLogAdapter; import org.springframework.boot.buildpack.platform.docker.DockerApi; import org.springframework.boot.buildpack.platform.docker.DockerApi.ContainerApi; import org.springframework.boot.buildpack.platform.docker.DockerApi.ImageApi; import org.springframework.boot.buildpack.platform.docker.DockerApi.VolumeApi; +import org.springframework.boot.buildpack.platform.docker.DockerLog; import org.springframework.boot.buildpack.platform.docker.TotalProgressPullListener; import org.springframework.boot.buildpack.platform.docker.configuration.DockerConfiguration; import org.springframework.boot.buildpack.platform.docker.transport.DockerEngineException; @@ -45,6 +47,7 @@ import static org.assertj.core.api.Assertions.assertThatExceptionOfType; import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; import static org.assertj.core.api.Assertions.assertThatIllegalStateException; +import static org.assertj.core.api.Assertions.assertThatNoException; import static org.mockito.ArgumentMatchers.any; import static org.mockito.ArgumentMatchers.eq; import static org.mockito.ArgumentMatchers.isNull; @@ -66,20 +69,37 @@ class BuilderTests { @Test void createWhenLogIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new Builder((BuildLog) null)) - .withMessage("Log must not be null"); + .withMessage("'log' must not be null"); } @Test void createWithDockerConfiguration() { + assertThatNoException().isThrownBy(() -> new Builder(BuildLog.toSystemOut())); + } + + @Test + void createDockerApiWithLogDockerLogDelegate() { Builder builder = new Builder(BuildLog.toSystemOut()); - assertThat(builder).isNotNull(); + assertThat(builder).extracting("docker") + .extracting("system") + .extracting("log") + .isInstanceOf(BuildLogAdapter.class); + } + + @Test + void createDockerApiWithLogDockerSystemOutDelegate() { + Builder builder = new Builder(mock(BuildLog.class)); + assertThat(builder).extracting("docker") + .extracting("system") + .extracting("log") + .isInstanceOf(DockerLog.toSystemOut().getClass()); } @Test void buildWhenRequestIsNullThrowsException() { Builder builder = new Builder(); assertThatIllegalArgumentException().isThrownBy(() -> builder.build(null)) - .withMessage("Request must not be null"); + .withMessage("'request' must not be null"); } @Test @@ -517,7 +537,7 @@ void buildWhenRequestedBuildpackNotInBuilderThrowsException() throws Exception { Builder builder = new Builder(BuildLog.to(out), docker, null); BuildpackReference reference = BuildpackReference.of("urn:cnb:builder:example/buildpack@1.2.3"); BuildRequest request = getTestRequest().withBuildpacks(reference); - assertThatIllegalArgumentException().isThrownBy(() -> builder.build(request)) + assertThatIllegalStateException().isThrownBy(() -> builder.build(request)) .withMessageContaining("'urn:cnb:builder:example/buildpack@1.2.3'") .withMessageContaining("not found in builder"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackCoordinatesTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackCoordinatesTests.java index 5a2820a85fcd..9341687a9938 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackCoordinatesTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackCoordinatesTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -99,7 +99,7 @@ void fromTomlWhenContainsBothStacksAndOrderThrowsException() throws IOException @Test void fromBuildpackMetadataWhenMetadataIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> BuildpackCoordinates.fromBuildpackMetadata(null)) - .withMessage("BuildpackMetadata must not be null"); + .withMessage("'buildpackMetadata' must not be null"); } @Test @@ -113,7 +113,7 @@ void fromBuildpackMetadataReturnsCoordinates() throws Exception { @Test void ofWhenIdIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> BuildpackCoordinates.of(null, null)) - .withMessage("ID must not be empty"); + .withMessage("'id' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackLayersMetadataTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackLayersMetadataTests.java index 31d692f849a6..6b53ff175624 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackLayersMetadataTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackLayersMetadataTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,6 +27,7 @@ import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; import static org.mockito.BDDMockito.given; import static org.mockito.Mockito.mock; @@ -54,14 +55,14 @@ void fromImageLoadsMetadata() throws IOException { @Test void fromImageWhenImageIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> BuildpackLayersMetadata.fromImage(null)) - .withMessage("Image must not be null"); + .withMessage("'image' must not be null"); } @Test void fromImageWhenImageConfigIsNullThrowsException() { Image image = mock(Image.class); assertThatIllegalArgumentException().isThrownBy(() -> BuildpackLayersMetadata.fromImage(image)) - .withMessage("ImageConfig must not be null"); + .withMessage("'imageConfig' must not be null"); } @Test @@ -70,7 +71,7 @@ void fromImageConfigWhenLabelIsMissingThrowsException() { ImageConfig imageConfig = mock(ImageConfig.class); given(image.getConfig()).willReturn(imageConfig); given(imageConfig.getLabels()).willReturn(Collections.singletonMap("alpha", "a")); - assertThatIllegalArgumentException().isThrownBy(() -> BuildpackLayersMetadata.fromImage(image)) + assertThatIllegalStateException().isThrownBy(() -> BuildpackLayersMetadata.fromImage(image)) .withMessage("No 'io.buildpacks.buildpack.layers' label found in image config labels 'alpha'"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackMetadataTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackMetadataTests.java index dd9fac9d1ae0..5916796df6dd 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackMetadataTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackMetadataTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,6 +27,7 @@ import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; import static org.mockito.BDDMockito.given; import static org.mockito.Mockito.mock; @@ -48,14 +49,14 @@ void fromImageLoadsMetadata() throws IOException { @Test void fromImageWhenImageIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> BuildpackMetadata.fromImage(null)) - .withMessage("Image must not be null"); + .withMessage("'image' must not be null"); } @Test void fromImageWhenImageConfigIsNullThrowsException() { Image image = mock(Image.class); assertThatIllegalArgumentException().isThrownBy(() -> BuildpackMetadata.fromImage(image)) - .withMessage("ImageConfig must not be null"); + .withMessage("'imageConfig' must not be null"); } @Test @@ -64,7 +65,7 @@ void fromImageConfigWhenLabelIsMissingThrowsException() { ImageConfig imageConfig = mock(ImageConfig.class); given(image.getConfig()).willReturn(imageConfig); given(imageConfig.getLabels()).willReturn(Collections.singletonMap("alpha", "a")); - assertThatIllegalArgumentException().isThrownBy(() -> BuildpackMetadata.fromImage(image)) + assertThatIllegalStateException().isThrownBy(() -> BuildpackMetadata.fromImage(image)) .withMessage("No 'io.buildpacks.buildpackage.metadata' label found in image config labels 'alpha'"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackReferenceTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackReferenceTests.java index 9fc08f4a13b6..b81b9c2cc625 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackReferenceTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/BuildpackReferenceTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,7 +33,7 @@ class BuildpackReferenceTests { @Test void ofWhenValueIsEmptyThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> BuildpackReference.of("")) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/DirectoryBuildpackTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/DirectoryBuildpackTests.java index 3fdbd1218596..aae2015c5780 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/DirectoryBuildpackTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/DirectoryBuildpackTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -37,7 +37,7 @@ import org.junit.jupiter.api.io.TempDir; import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; import static org.assertj.core.api.Assertions.tuple; import static org.mockito.Mockito.mock; @@ -89,8 +89,7 @@ void resolveWhenFileUrl() throws Exception { void resolveWhenDirectoryWithoutBuildpackTomlThrowsException() throws Exception { Files.createDirectories(this.buildpackDir.toPath()); BuildpackReference reference = BuildpackReference.of(this.buildpackDir.toString()); - assertThatIllegalArgumentException() - .isThrownBy(() -> DirectoryBuildpack.resolve(this.resolverContext, reference)) + assertThatIllegalStateException().isThrownBy(() -> DirectoryBuildpack.resolve(this.resolverContext, reference)) .withMessageContaining("Buildpack descriptor 'buildpack.toml' is required") .withMessageContaining(this.buildpackDir.getAbsolutePath()); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/ImageBuildpackTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/ImageBuildpackTests.java index 4385e419300c..78dba090070c 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/ImageBuildpackTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/ImageBuildpackTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,6 +43,7 @@ import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; import static org.assertj.core.api.Assertions.fail; import static org.assertj.core.api.Assertions.tuple; import static org.mockito.ArgumentMatchers.any; @@ -156,7 +157,7 @@ void resolveWhenMissingMetadataLabelThrowsException() throws Exception { BuildpackResolverContext resolverContext = mock(BuildpackResolverContext.class); given(resolverContext.fetchImage(any(), any())).willReturn(image); BuildpackReference reference = BuildpackReference.of("docker://example/buildpack1:latest"); - assertThatIllegalArgumentException().isThrownBy(() -> ImageBuildpack.resolve(resolverContext, reference)) + assertThatIllegalStateException().isThrownBy(() -> ImageBuildpack.resolve(resolverContext, reference)) .withMessageContaining("No 'io.buildpacks.buildpackage.metadata' label found"); } @@ -165,7 +166,7 @@ void resolveWhenFullyQualifiedReferenceWithInvalidImageReferenceThrowsException( BuildpackReference reference = BuildpackReference.of("docker://buildpack@0.0.1"); BuildpackResolverContext resolverContext = mock(BuildpackResolverContext.class); assertThatIllegalArgumentException().isThrownBy(() -> ImageBuildpack.resolve(resolverContext, reference)) - .withMessageContaining("Unable to parse image reference \"buildpack@0.0.1\""); + .withMessageContaining("'value' [buildpack@0.0.1] must be an image reference"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/LifecycleVersionTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/LifecycleVersionTests.java index 4bd256bb7506..869fcfbfc08b 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/LifecycleVersionTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/LifecycleVersionTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,19 +31,19 @@ class LifecycleVersionTests { @Test void parseWhenValueIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> LifecycleVersion.parse(null)) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test void parseWhenTooLongThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> LifecycleVersion.parse("v1.2.3.4")) - .withMessage("Malformed version number '1.2.3.4'"); + .withMessage("'value' [v1.2.3.4] must be a valid version number"); } @Test void parseWhenNonNumericThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> LifecycleVersion.parse("v1.2.3a")) - .withMessage("Malformed version number '1.2.3a'"); + .withMessage("'value' [v1.2.3a] must be a valid version number"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/StackIdTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/StackIdTests.java index a804b596a317..70f075b8cd22 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/StackIdTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/build/StackIdTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ class StackIdTests { @Test void fromImageWhenImageIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> StackId.fromImage(null)) - .withMessage("Image must not be null"); + .withMessage("'image' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/DockerApiTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/DockerApiTests.java index 2f717e44b620..4a17fcc30b60 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/DockerApiTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/DockerApiTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,10 +23,7 @@ import java.io.OutputStream; import java.net.URI; import java.net.URISyntaxException; -import java.nio.file.Path; -import java.util.ArrayList; import java.util.Arrays; -import java.util.List; import org.apache.commons.compress.archivers.tar.TarArchiveEntry; import org.apache.commons.compress.archivers.tar.TarArchiveInputStream; @@ -62,6 +59,8 @@ import org.springframework.boot.buildpack.platform.io.IOConsumer; import org.springframework.boot.buildpack.platform.io.Owner; import org.springframework.boot.buildpack.platform.io.TarArchive; +import org.springframework.boot.testsupport.system.CapturedOutput; +import org.springframework.boot.testsupport.system.OutputCaptureExtension; import org.springframework.util.LinkedMultiValueMap; import org.springframework.util.MultiValueMap; @@ -85,7 +84,7 @@ * @author Rafael Ceccone * @author Moritz Halbritter */ -@ExtendWith(MockitoExtension.class) +@ExtendWith({ MockitoExtension.class, OutputCaptureExtension.class }) class DockerApiTests { private static final String API_URL = "/v" + DockerApi.API_VERSION; @@ -111,7 +110,7 @@ class DockerApiTests { @BeforeEach void setup() { - this.dockerApi = new DockerApi(this.http); + this.dockerApi = new DockerApi(this.http, DockerLog.toSystemOut()); } private HttpTransport http() { @@ -194,14 +193,14 @@ void setup() { @Test void pullWhenReferenceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.pull(null, null, this.pullListener)) - .withMessage("Reference must not be null"); + .withMessage("'reference' must not be null"); } @Test void pullWhenListenerIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> this.api.pull(ImageReference.of("ubuntu"), null, null)) - .withMessage("Listener must not be null"); + .withMessage("'listener' must not be null"); } @Test @@ -267,14 +266,14 @@ void pullWithPlatformAndInsufficientApiVersionThrowsException() throws Exception @Test void pushWhenReferenceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.push(null, this.pushListener, null)) - .withMessage("Reference must not be null"); + .withMessage("'reference' must not be null"); } @Test void pushWhenListenerIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> this.api.push(ImageReference.of("ubuntu"), null, null)) - .withMessage("Listener must not be null"); + .withMessage("'listener' must not be null"); } @Test @@ -302,14 +301,14 @@ void pushWithErrorInStreamThrowsException() throws Exception { @Test void loadWhenArchiveIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.load(null, UpdateListener.none())) - .withMessage("Archive must not be null"); + .withMessage("'archive' must not be null"); } @Test void loadWhenListenerIsNullThrowsException() { ImageArchive archive = mock(ImageArchive.class); assertThatIllegalArgumentException().isThrownBy(() -> this.api.load(archive, null)) - .withMessage("Listener must not be null"); + .withMessage("'listener' must not be null"); } @Test // gh-23130 @@ -352,7 +351,7 @@ void loadLoadsImage() throws Exception { @Test void removeWhenReferenceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.remove(null, true)) - .withMessage("Reference must not be null"); + .withMessage("'reference' must not be null"); } @Test @@ -380,7 +379,7 @@ void removeWhenForceIsTrueRemovesContainer() throws Exception { @Test void inspectWhenReferenceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.inspect(null)) - .withMessage("Reference must not be null"); + .withMessage("'reference' must not be null"); } @Test @@ -392,21 +391,6 @@ void inspectInspectImage() throws Exception { assertThat(image.getLayers()).hasSize(46); } - @Test - @SuppressWarnings("removal") - void exportLayersWhenReferenceIsNullThrowsException() { - assertThatIllegalArgumentException().isThrownBy(() -> this.api.exportLayerFiles(null, (name, archive) -> { - })).withMessage("Reference must not be null"); - } - - @Test - @SuppressWarnings("removal") - void exportLayersWhenExportsIsNullThrowsException() { - ImageReference reference = ImageReference.of("gcr.io/paketo-buildpacks/builder:base"); - assertThatIllegalArgumentException().isThrownBy(() -> this.api.exportLayerFiles(reference, null)) - .withMessage("Exports must not be null"); - } - @Test void exportLayersExportsLayerTars() throws Exception { ImageReference reference = ImageReference.of("gcr.io/paketo-buildpacks/builder:base"); @@ -463,41 +447,18 @@ void exportLayersWithSymlinksExportsLayerTars() throws Exception { .containsExactly("/cnb/stack.toml"); } - @Test - @SuppressWarnings("removal") - void exportLayerFilesDeletesTempFiles() throws Exception { - ImageReference reference = ImageReference.of("gcr.io/paketo-buildpacks/builder:base"); - URI exportUri = new URI(IMAGES_URL + "/gcr.io/paketo-buildpacks/builder:base/get"); - given(DockerApiTests.this.http.get(exportUri)).willReturn(responseOf("export.tar")); - List layerFilePaths = new ArrayList<>(); - this.api.exportLayerFiles(reference, (name, path) -> layerFilePaths.add(path)); - layerFilePaths.forEach((path) -> assertThat(path.toFile()).doesNotExist()); - } - - @Test - @SuppressWarnings("removal") - void exportLayersWithNoManifestThrowsException() throws Exception { - ImageReference reference = ImageReference.of("gcr.io/paketo-buildpacks/builder:base"); - URI exportUri = new URI(IMAGES_URL + "/gcr.io/paketo-buildpacks/builder:base/get"); - given(DockerApiTests.this.http.get(exportUri)).willReturn(responseOf("export-no-manifest.tar")); - String expectedMessage = "Exported image '%s' does not contain 'index.json' or 'manifest.json'" - .formatted(reference); - assertThatIllegalStateException().isThrownBy(() -> this.api.exportLayerFiles(reference, (name, archive) -> { - })).withMessageContaining(expectedMessage); - } - @Test void tagWhenReferenceIsNullThrowsException() { ImageReference tag = ImageReference.of("localhost:5000/ubuntu"); assertThatIllegalArgumentException().isThrownBy(() -> this.api.tag(null, tag)) - .withMessage("SourceReference must not be null"); + .withMessage("'sourceReference' must not be null"); } @Test void tagWhenTargetIsNullThrowsException() { ImageReference reference = ImageReference.of("localhost:5000/ubuntu"); assertThatIllegalArgumentException().isThrownBy(() -> this.api.tag(reference, null)) - .withMessage("TargetReference must not be null"); + .withMessage("'targetReference' must not be null"); } @Test @@ -541,7 +502,7 @@ void setup() { @Test void createWhenConfigIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.create(null, null)) - .withMessage("Config must not be null"); + .withMessage("'config' must not be null"); } @Test @@ -628,7 +589,7 @@ void createWithPlatformAndKnownInsufficientApiVersionThrowsException() throws Ex @Test void startWhenReferenceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.start(null)) - .withMessage("Reference must not be null"); + .withMessage("'reference' must not be null"); } @Test @@ -643,14 +604,14 @@ void startStartsContainer() throws Exception { @Test void logsWhenReferenceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.logs(null, UpdateListener.none())) - .withMessage("Reference must not be null"); + .withMessage("'reference' must not be null"); } @Test void logsWhenListenerIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> this.api.logs(ContainerReference.of("e90e34656806"), null)) - .withMessage("Listener must not be null"); + .withMessage("'listener' must not be null"); } @Test @@ -668,7 +629,7 @@ void logsProducesEvents() throws Exception { @Test void waitWhenReferenceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.wait(null)) - .withMessage("Reference must not be null"); + .withMessage("'reference' must not be null"); } @Test @@ -683,7 +644,7 @@ void waitReturnsStatus() throws Exception { @Test void removeWhenReferenceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.remove(null, true)) - .withMessage("Reference must not be null"); + .withMessage("'reference' must not be null"); } @Test @@ -719,7 +680,7 @@ void setup() { @Test void deleteWhenNameIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> this.api.delete(null, false)) - .withMessage("Name must not be null"); + .withMessage("'name' must not be null"); } @Test @@ -773,9 +734,10 @@ void getApiVersionWithNoVersionHeaderReturnsUnknownVersion() throws Exception { } @Test - void getApiVersionWithExceptionReturnsUnknownVersion() throws Exception { + void getApiVersionWithExceptionReturnsUnknownVersion(CapturedOutput output) throws Exception { given(http().head(eq(new URI(PING_URL)))).willThrow(new IOException("simulated error")); assertThat(this.api.getApiVersion()).isEqualTo(DockerApi.UNKNOWN_API_VERSION); + assertThat(output).contains("Warning: Failed to determine Docker API version: simulated error"); } } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/DockerLogTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/DockerLogTests.java new file mode 100644 index 000000000000..6e9ec2afe42f --- /dev/null +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/DockerLogTests.java @@ -0,0 +1,51 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.buildpack.platform.docker; + +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; + +import org.springframework.boot.testsupport.system.CapturedOutput; +import org.springframework.boot.testsupport.system.OutputCaptureExtension; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Tests for {@link DockerLog}. + * + * @author Dmytro nosan + */ +@ExtendWith(OutputCaptureExtension.class) +class DockerLogTests { + + @Test + void toSystemOutPrintsToSystemOut(CapturedOutput output) { + DockerLog logger = DockerLog.toSystemOut(); + logger.log("Hello world"); + assertThat(output.getErr()).isEmpty(); + assertThat(output.getOut()).contains("Hello world"); + } + + @Test + void toPrintsToOutput(CapturedOutput output) { + DockerLog logger = DockerLog.to(System.err); + logger.log("Hello world"); + assertThat(output.getOut()).isEmpty(); + assertThat(output.getErr()).contains("Hello world"); + } + +} diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/ProgressUpdateEventTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/ProgressUpdateEventTests.java index dd43af828992..6044f6fe0f27 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/ProgressUpdateEventTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/ProgressUpdateEventTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,20 +39,14 @@ void getStatusReturnsStatus() { } @Test - @SuppressWarnings("removal") - void getProgressDetailsReturnsProgressDetails() { + void getProgressDetailReturnsProgressDetails() { ProgressUpdateEvent event = createEvent(); - assertThat(event.getProgressDetail().getCurrent()).isOne(); - assertThat(event.getProgressDetail().getTotal()).isEqualTo(2); assertThat(event.getProgressDetail().asPercentage()).isEqualTo(50); } @Test - @SuppressWarnings("removal") - void getProgressDetailsReturnsProgressDetailsForLongNumbers() { + void getProgressDetailReturnsProgressDetailsForLongNumbers() { ProgressUpdateEvent event = createEvent("status", new ProgressDetail(4000000000L, 8000000000L), "progress"); - assertThat(event.getProgressDetail().getCurrent()).isEqualTo(Integer.MAX_VALUE); - assertThat(event.getProgressDetail().getTotal()).isEqualTo(Integer.MAX_VALUE); assertThat(event.getProgressDetail().asPercentage()).isEqualTo(50); } @@ -62,27 +56,6 @@ void getProgressReturnsProgress() { assertThat(event.getProgress()).isEqualTo("progress"); } - @Test - @SuppressWarnings("removal") - void progressDetailIsEmptyWhenCurrentIsNullReturnsTrue() { - ProgressDetail detail = new ProgressDetail(null, 2L); - assertThat(ProgressDetail.isEmpty(detail)).isTrue(); - } - - @Test - @SuppressWarnings("removal") - void progressDetailIsEmptyWhenTotalIsNullReturnsTrue() { - ProgressDetail detail = new ProgressDetail(1L, null); - assertThat(ProgressDetail.isEmpty(detail)).isTrue(); - } - - @Test - @SuppressWarnings("removal") - void progressDetailIsEmptyWhenTotalAndCurrentAreNotNullReturnsFalse() { - ProgressDetail detail = new ProgressDetail(1L, 2L); - assertThat(ProgressDetail.isEmpty(detail)).isFalse(); - } - protected E createEvent() { return createEvent("status", new ProgressDetail(1L, 2L), "progress"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/PullUpdateEventTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/PullUpdateEventTests.java index 7c95f9bcbd0f..7e762003cdda 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/PullUpdateEventTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/PullUpdateEventTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,8 +36,6 @@ void readValueWhenFullDeserializesJson() throws Exception { PullImageUpdateEvent.class); assertThat(event.getId()).isEqualTo("4f4fb700ef54"); assertThat(event.getStatus()).isEqualTo("Extracting"); - assertThat(event.getProgressDetail().getCurrent()).isEqualTo(16); - assertThat(event.getProgressDetail().getTotal()).isEqualTo(32); assertThat(event.getProgressDetail().asPercentage()).isEqualTo(50); assertThat(event.getProgress()).isEqualTo("[==================================================>] 32B/32B"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/TotalProgressEventTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/TotalProgressEventTests.java index 02f736671388..202eb2edaa38 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/TotalProgressEventTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/TotalProgressEventTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,13 +38,13 @@ void create() { @Test void createWhenPercentLessThanZeroThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new TotalProgressEvent(-1)) - .withMessage("Percent must be in the range 0 to 100"); + .withMessage("'percent' must be in the range 0 to 100"); } @Test void createWhenEventMoreThanOneHundredThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new TotalProgressEvent(101)) - .withMessage("Percent must be in the range 0 to 100"); + .withMessage("'percent' must be in the range 0 to 100"); } } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/DockerConnectionExceptionTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/DockerConnectionExceptionTests.java index 68c5546e1a98..48c73b22ba72 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/DockerConnectionExceptionTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/DockerConnectionExceptionTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,13 +35,13 @@ class DockerConnectionExceptionTests { @Test void createWhenHostIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new DockerConnectionException(null, null)) - .withMessage("Host must not be null"); + .withMessage("'host' must not be null"); } @Test void createWhenCauseIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new DockerConnectionException(HOST, null)) - .withMessage("Cause must not be null"); + .withMessage("'cause' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/DockerEngineExceptionTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/DockerEngineExceptionTests.java index 72c3ff03213f..87cadc062bbe 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/DockerEngineExceptionTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/DockerEngineExceptionTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -57,14 +57,14 @@ class DockerEngineExceptionTests { void createWhenHostIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new DockerEngineException(null, null, 404, null, NO_ERRORS, NO_MESSAGE)) - .withMessage("Host must not be null"); + .withMessage("'host' must not be null"); } @Test void createWhenUriIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new DockerEngineException(HOST, null, 404, null, NO_ERRORS, NO_MESSAGE)) - .withMessage("URI must not be null"); + .withMessage("'uri' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/RemoteHttpClientTransportTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/RemoteHttpClientTransportTests.java index 529709d5cc38..87767ed49ac2 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/RemoteHttpClientTransportTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/transport/RemoteHttpClientTransportTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,7 +28,7 @@ import org.springframework.boot.buildpack.platform.docker.ssl.SslContextFactory; import static org.assertj.core.api.Assertions.assertThat; -import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; +import static org.assertj.core.api.Assertions.assertThatIllegalStateException; import static org.mockito.BDDMockito.given; import static org.mockito.Mockito.mock; @@ -92,7 +92,7 @@ void createIfPossibleWhenTlsVerifyUsesHttps() throws Exception { void createIfPossibleWhenTlsVerifyWithMissingCertPathThrowsException() { ResolvedDockerHost dockerHost = ResolvedDockerHost .from(DockerHostConfiguration.forAddress("tcp://192.168.1.2:2376", true, null)); - assertThatIllegalArgumentException().isThrownBy(() -> RemoteHttpClientTransport.createIfPossible(dockerHost)) + assertThatIllegalStateException().isThrownBy(() -> RemoteHttpClientTransport.createIfPossible(dockerHost)) .withMessageContaining("Docker host TLS verification requires trust material"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ApiVersionTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ApiVersionTests.java index d06d315eebf8..f3339d6cf7b6 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ApiVersionTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ApiVersionTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,19 +34,19 @@ class ApiVersionTests { @Test void parseWhenVersionIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ApiVersion.parse(null)) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test void parseWhenVersionIsEmptyThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ApiVersion.parse("")) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test void parseWhenVersionDoesNotMatchPatternThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ApiVersion.parse("bad")) - .withMessage("Malformed version number 'bad'"); + .withMessage("'value' [bad] must contain a well formed version number"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/BindingTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/BindingTests.java index 7f8d29c269f5..ebd7e1c3de03 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/BindingTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/BindingTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -41,7 +41,7 @@ void ofReturnsValue() { @Test void ofWithNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> Binding.of(null)) - .withMessageContaining("Value must not be null"); + .withMessageContaining("'value' must not be null"); } @Test @@ -53,13 +53,13 @@ void fromReturnsValue() { @Test void fromWithNullSourceThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> Binding.from((String) null, "container-dest")) - .withMessageContaining("Source must not be null"); + .withMessageContaining("'source' must not be null"); } @Test void fromWithNullDestinationThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> Binding.from("host-src", null)) - .withMessageContaining("Destination must not be null"); + .withMessageContaining("'destination' must not be null"); } @Test @@ -71,7 +71,7 @@ void fromVolumeNameSourceReturnsValue() { @Test void fromVolumeNameSourceWithNullSourceThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> Binding.from((VolumeName) null, "container-dest")) - .withMessageContaining("SourceVolume must not be null"); + .withMessageContaining("'sourceVolume' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerConfigTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerConfigTests.java index ce174809f0a7..c59f70926201 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerConfigTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerConfigTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,14 +39,14 @@ class ContainerConfigTests extends AbstractJsonTests { @Test void ofWhenImageReferenceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ContainerConfig.of(null, (update) -> { - })).withMessage("ImageReference must not be null"); + })).withMessage("'imageReference' must not be null"); } @Test void ofWhenUpdateIsNullThrowsException() { ImageReference imageReference = ImageReference.of("ubuntu:bionic"); assertThatIllegalArgumentException().isThrownBy(() -> ContainerConfig.of(imageReference, null)) - .withMessage("Update must not be null"); + .withMessage("'update' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerContentTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerContentTests.java index e6027da59244..285ec9b7e148 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerContentTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerContentTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,21 +34,21 @@ class ContainerContentTests { @Test void ofWhenArchiveIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ContainerContent.of(null)) - .withMessage("Archive must not be null"); + .withMessage("'archive' must not be null"); } @Test void ofWhenDestinationPathIsNullThrowsException() { TarArchive archive = mock(TarArchive.class); assertThatIllegalArgumentException().isThrownBy(() -> ContainerContent.of(archive, null)) - .withMessage("DestinationPath must not be empty"); + .withMessage("'destinationPath' must not be empty"); } @Test void ofWhenDestinationPathIsEmptyThrowsException() { TarArchive archive = mock(TarArchive.class); assertThatIllegalArgumentException().isThrownBy(() -> ContainerContent.of(archive, "")) - .withMessage("DestinationPath must not be empty"); + .withMessage("'destinationPath' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerReferenceTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerReferenceTests.java index 41d47672348b..e6aa88b6eded 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerReferenceTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ContainerReferenceTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,13 +38,13 @@ void ofCreatesInstance() { @Test void ofWhenNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ContainerReference.of(null)) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test void ofWhenEmptyThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ContainerReference.of("")) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImageNameTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImageNameTests.java index 717d165d406d..43c4aacc8d1d 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImageNameTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImageNameTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -118,25 +118,26 @@ void ofWhenLegacyDomainUsesNewDomain() { @Test void ofWhenNameIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of(null)) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test void ofWhenNameIsEmptyThrowsException() { - assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of("")).withMessage("Value must not be empty"); + assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of("")) + .withMessage("'value' must not be empty"); } @Test void ofWhenContainsUppercaseThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of("Test")) - .withMessageContaining("Unable to parse name") + .withMessageContaining("must be a parsable name") .withMessageContaining("Test"); } @Test void ofWhenNameIncludesTagThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ImageName.of("ubuntu:latest")) - .withMessageContaining("Unable to parse name") + .withMessageContaining("must be a parsable name") .withMessageContaining(":latest"); } @@ -144,7 +145,7 @@ void ofWhenNameIncludesTagThrowsException() { void ofWhenNameIncludeDigestThrowsException() { assertThatIllegalArgumentException().isThrownBy( () -> ImageName.of("ubuntu@sha256:47bfdb88c3ae13e488167607973b7688f69d9e8c142c2045af343ec199649c09")) - .withMessageContaining("Unable to parse name") + .withMessageContaining("must be a parsable name") .withMessageContaining("@sha256:47b"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImagePlatformTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImagePlatformTests.java index 46da80ad2399..b452f4cd95c3 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImagePlatformTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImagePlatformTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,13 +48,13 @@ void ofWithOsAndArchitectureAndVariantParses() { @Test void ofWithEmptyValueFails() { assertThatIllegalArgumentException().isThrownBy(() -> ImagePlatform.of("")) - .withMessageContaining("Value must not be empty"); + .withMessageContaining("'value' must not be empty"); } @Test void ofWithTooManySegmentsFails() { assertThatIllegalArgumentException().isThrownBy(() -> ImagePlatform.of("linux/amd64/v1/extra")) - .withMessageContaining("value 'linux/amd64/v1/extra'"); + .withMessageContaining("'value' [linux/amd64/v1/extra] must be in the form"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImageReferenceTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImageReferenceTests.java index b22c27e1ba2c..04e7b92f2d18 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImageReferenceTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/ImageReferenceTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -189,7 +189,7 @@ void ofWhenHasIllegalCharacterThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> ImageReference .of("registry.example.com/example/example-app:1.6.0-dev.2.uncommitted+wip.foo.c75795d")) - .withMessageContaining("Unable to parse image reference"); + .withMessageContaining("must be an image reference"); } @Test @@ -197,7 +197,7 @@ void ofWhenContainsUpperCaseThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> ImageReference .of("europe-west1-docker.pkg.dev/aaaaaa-bbbbb-123456/docker-registry/bootBuildImage:0.0.1")) - .withMessageContaining("Unable to parse image reference"); + .withMessageContaining("must be an image reference"); } @Test @@ -205,7 +205,7 @@ void ofWhenContainsUpperCaseThrowsException() { void ofWhenIsVeryLongAndHasIllegalCharacter() { assertThatIllegalArgumentException().isThrownBy(() -> ImageReference .of("docker.io/library/this-image-has-a-long-name-with-an-invalid-tag-which-is-at-danger-of-catastrophic-backtracking:1.0.0+1234")) - .withMessageContaining("Unable to parse image reference"); + .withMessageContaining("must be an image reference"); } @Test @@ -248,7 +248,7 @@ void randomWithLengthGeneratesRandomName() { @Test void randomWherePrefixIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> ImageReference.random(null)) - .withMessage("Prefix must not be null"); + .withMessage("'prefix' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/LayerIdTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/LayerIdTests.java index d910d0f8f93e..3317665b8937 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/LayerIdTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/LayerIdTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -52,12 +52,12 @@ void hashCodeAndEquals() { @Test void ofWhenValueIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> LayerId.of((String) null)) - .withMessage("Value must not be empty"); + .withMessage("'value' must not be empty"); } @Test void ofWhenValueIsEmptyThrowsException() { - assertThatIllegalArgumentException().isThrownBy(() -> LayerId.of(" ")).withMessage("Value must not be empty"); + assertThatIllegalArgumentException().isThrownBy(() -> LayerId.of(" ")).withMessage("'value' must not be empty"); } @Test @@ -80,13 +80,13 @@ void ofSha256DigestWithZeroPadding() { @Test void ofSha256DigestWhenNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> LayerId.ofSha256Digest((byte[]) null)) - .withMessage("Digest must not be null"); + .withMessage("'digest' must not be null"); } @Test void ofSha256DigestWhenWrongLengthThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> LayerId.ofSha256Digest(new byte[31])) - .withMessage("Digest must be exactly 32 bytes"); + .withMessage("'digest' must be exactly 32 bytes"); } } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/LayerTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/LayerTests.java index bad2124fb8d5..1d46b6fedffd 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/LayerTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/LayerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,13 +40,13 @@ class LayerTests { @Test void ofWhenLayoutIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> Layer.of((IOConsumer) null)) - .withMessage("Layout must not be null"); + .withMessage("'layout' must not be null"); } @Test void fromTarArchiveWhenTarArchiveIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> Layer.fromTarArchive(null)) - .withMessage("TarArchive must not be null"); + .withMessage("'tarArchive' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/RandomStringTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/RandomStringTests.java index aa9284485d92..63d3c4e295cb 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/RandomStringTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/RandomStringTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ class RandomStringTests { @Test void generateWhenPrefixIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> RandomString.generate(null, 10)) - .withMessage("Prefix must not be null"); + .withMessage("'prefix' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/VolumeNameTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/VolumeNameTests.java index 90bafa796e56..45694928cc70 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/VolumeNameTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/docker/type/VolumeNameTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ class VolumeNameTests { @Test void randomWhenPrefixIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> VolumeName.random(null)) - .withMessage("Prefix must not be null"); + .withMessage("'prefix' must not be null"); } @Test @@ -57,25 +57,25 @@ void randomStringWithLengthGeneratesRandomString() { @Test void basedOnWhenSourceIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> VolumeName.basedOn(null, "prefix", "suffix", 6)) - .withMessage("Source must not be null"); + .withMessage("'source' must not be null"); } @Test void basedOnWhenNameExtractorIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> VolumeName.basedOn("test", null, "prefix", "suffix", 6)) - .withMessage("NameExtractor must not be null"); + .withMessage("'nameExtractor' must not be null"); } @Test void basedOnWhenPrefixIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> VolumeName.basedOn("test", null, "suffix", 6)) - .withMessage("Prefix must not be null"); + .withMessage("'prefix' must not be null"); } @Test void basedOnWhenSuffixIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> VolumeName.basedOn("test", "prefix", null, 6)) - .withMessage("Suffix must not be null"); + .withMessage("'suffix' must not be null"); } @Test @@ -87,13 +87,13 @@ void basedOnGeneratesHashBasedName() { @Test void basedOnWhenSizeIsTooBigThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> VolumeName.basedOn("name", "prefix", "suffix", 33)) - .withMessage("DigestLength must be less than or equal to 32"); + .withMessage("'digestLength' must be less than or equal to 32"); } @Test void ofWhenValueIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> VolumeName.of(null)) - .withMessage("Value must not be null"); + .withMessage("'value' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/ContentTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/ContentTests.java index cb9a8eb1f20a..bb2c5180d9dc 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/ContentTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/ContentTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -35,9 +35,9 @@ class ContentTests { @Test - void ofWhenStreamIsNullThrowsException() { + void ofWhenSupplierIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> Content.of(1, (IOSupplier) null)) - .withMessage("Supplier must not be null"); + .withMessage("'supplier' must not be null"); } @Test @@ -51,7 +51,7 @@ void ofWhenStreamReturnsWritable() throws Exception { @Test void ofWhenStringIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> Content.of((String) null)) - .withMessage("String must not be null"); + .withMessage("'string' must not be null"); } @Test @@ -63,7 +63,7 @@ void ofWhenStringReturnsWritable() throws Exception { @Test void ofWhenBytesIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> Content.of((byte[]) null)) - .withMessage("Bytes must not be null"); + .withMessage("'bytes' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/InspectedContentTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/InspectedContentTests.java index dfbf2483ba0f..f38ec190727a 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/InspectedContentTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/InspectedContentTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,19 +39,19 @@ class InspectedContentTests { @Test void ofWhenInputStreamThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> InspectedContent.of((InputStream) null)) - .withMessage("InputStream must not be null"); + .withMessage("'inputStream' must not be null"); } @Test void ofWhenContentIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> InspectedContent.of((Content) null)) - .withMessage("Content must not be null"); + .withMessage("'content' must not be null"); } @Test void ofWhenConsumerIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> InspectedContent.of((IOConsumer) null)) - .withMessage("Writer must not be null"); + .withMessage("'writer' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/ZipFileTarArchiveTests.java b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/ZipFileTarArchiveTests.java index 48c2f6f70e11..58b49b645f26 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/ZipFileTarArchiveTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-buildpack-platform/src/test/java/org/springframework/boot/buildpack/platform/io/ZipFileTarArchiveTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -46,7 +46,7 @@ class ZipFileTarArchiveTests { @Test void createWhenZipIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new ZipFileTarArchive(null, Owner.ROOT)) - .withMessage("Zip must not be null"); + .withMessage("'zip' must not be null"); } @Test @@ -54,7 +54,7 @@ void createWhenOwnerIsNullThrowsException() throws Exception { File file = new File(this.tempDir, "test.zip"); writeTestZip(file); assertThatIllegalArgumentException().isThrownBy(() -> new ZipFileTarArchive(file, null)) - .withMessage("Owner must not be null"); + .withMessage("'owner' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-cli/build.gradle b/spring-boot-project/spring-boot-tools/spring-boot-cli/build.gradle index 81df1684ca72..3ec491f89f80 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-cli/build.gradle +++ b/spring-boot-project/spring-boot-tools/spring-boot-cli/build.gradle @@ -11,7 +11,6 @@ plugins { description = "Spring Boot CLI" configurations { - dependenciesBom loader testRepository compileOnlyProject @@ -21,8 +20,6 @@ configurations { dependencies { compileOnlyProject(project(":spring-boot-project:spring-boot")) - dependenciesBom(project(path: ":spring-boot-project:spring-boot-dependencies", configuration: "effectiveBom")) - implementation(project(":spring-boot-project:spring-boot-tools:spring-boot-loader-tools")) implementation("com.vaadin.external.google:android-json") implementation("jline:jline") diff --git a/spring-boot-project/spring-boot-tools/spring-boot-cli/src/main/java/org/springframework/boot/cli/command/CommandRunner.java b/spring-boot-project/spring-boot-tools/spring-boot-cli/src/main/java/org/springframework/boot/cli/command/CommandRunner.java index 3360b402d9ef..4a79dc08ec81 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-cli/src/main/java/org/springframework/boot/cli/command/CommandRunner.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-cli/src/main/java/org/springframework/boot/cli/command/CommandRunner.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -72,7 +72,7 @@ public String getName() { * @param commands the commands to add */ public void addCommands(Iterable commands) { - Assert.notNull(commands, "Commands must not be null"); + Assert.notNull(commands, "'commands' must not be null"); for (Command command : commands) { addCommand(command); } @@ -83,7 +83,7 @@ public void addCommands(Iterable commands) { * @param command the command to add. */ public void addCommand(Command command) { - Assert.notNull(command, "Command must not be null"); + Assert.notNull(command, "'command' must not be null"); this.commands.add(command); } @@ -95,7 +95,7 @@ public void addCommand(Command command) { * @see #isOptionCommand(Command) */ public void setOptionCommands(Class... commandClasses) { - Assert.notNull(commandClasses, "CommandClasses must not be null"); + Assert.notNull(commandClasses, "'commandClasses' must not be null"); this.optionCommandClasses = commandClasses; } @@ -105,7 +105,7 @@ public void setOptionCommands(Class... commandClasses) { * @param commandClasses the classes of hidden commands */ public void setHiddenCommands(Class... commandClasses) { - Assert.notNull(commandClasses, "CommandClasses must not be null"); + Assert.notNull(commandClasses, "'commandClasses' must not be null"); this.hiddenCommandClasses = commandClasses; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-cli/src/main/java/org/springframework/boot/cli/command/init/InitCommand.java b/spring-boot-project/spring-boot-tools/spring-boot-cli/src/main/java/org/springframework/boot/cli/command/init/InitCommand.java index 557c44304d7c..0e20291d33ed 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-cli/src/main/java/org/springframework/boot/cli/command/init/InitCommand.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-cli/src/main/java/org/springframework/boot/cli/command/init/InitCommand.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -220,7 +220,7 @@ protected void generateProject(OptionSet options) throws IOException { protected ProjectGenerationRequest createProjectGenerationRequest(OptionSet options) { List nonOptionArguments = new ArrayList(options.nonOptionArguments()); - Assert.isTrue(nonOptionArguments.size() <= 1, "Only the target location may be specified"); + Assert.state(nonOptionArguments.size() <= 1, "Only the target location may be specified"); ProjectGenerationRequest request = new ProjectGenerationRequest(); request.setServiceUrl(options.valueOf(this.target)); if (options.has(this.bootVersion)) { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/ConfigurationMetadataAnnotationProcessor.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/ConfigurationMetadataAnnotationProcessor.java index e44bd5e54f60..4e18cb466594 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/ConfigurationMetadataAnnotationProcessor.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/ConfigurationMetadataAnnotationProcessor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,6 +48,7 @@ import org.springframework.boot.configurationprocessor.metadata.ConfigurationMetadata; import org.springframework.boot.configurationprocessor.metadata.InvalidConfigurationMetadataException; import org.springframework.boot.configurationprocessor.metadata.ItemDeprecation; +import org.springframework.boot.configurationprocessor.metadata.ItemIgnore; import org.springframework.boot.configurationprocessor.metadata.ItemMetadata; /** @@ -372,6 +373,7 @@ private String getPrefix(AnnotationMirror annotation) { protected ConfigurationMetadata writeMetadata() throws Exception { ConfigurationMetadata metadata = this.metadataCollector.getMetadata(); metadata = mergeAdditionalMetadata(metadata); + removeIgnored(metadata); if (!metadata.getItems().isEmpty()) { this.metadataStore.writeMetadata(metadata); return metadata; @@ -379,6 +381,12 @@ protected ConfigurationMetadata writeMetadata() throws Exception { return null; } + private void removeIgnored(ConfigurationMetadata metadata) { + for (ItemIgnore itemIgnore : metadata.getIgnored()) { + metadata.removeMetadata(itemIgnore.getType(), itemIgnore.getName()); + } + } + private ConfigurationMetadata mergeAdditionalMetadata(ConfigurationMetadata metadata) { try { ConfigurationMetadata merged = new ConfigurationMetadata(metadata); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/ConfigurationMetadata.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/ConfigurationMetadata.java index 3b8c1fd2785a..d8085738bc90 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/ConfigurationMetadata.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/ConfigurationMetadata.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,6 +22,7 @@ import java.util.List; import java.util.Map; +import org.springframework.boot.configurationprocessor.metadata.ItemMetadata.ItemType; import org.springframework.boot.configurationprocessor.support.ConventionUtils; /** @@ -29,6 +30,7 @@ * * @author Stephane Nicoll * @author Phillip Webb + * @author Moritz Halbritter * @since 1.2.0 * @see ItemMetadata */ @@ -38,14 +40,18 @@ public class ConfigurationMetadata { private final Map> hints; + private final Map> ignored; + public ConfigurationMetadata() { this.items = new LinkedHashMap<>(); this.hints = new LinkedHashMap<>(); + this.ignored = new LinkedHashMap<>(); } public ConfigurationMetadata(ConfigurationMetadata metadata) { this.items = new LinkedHashMap<>(metadata.items); this.hints = new LinkedHashMap<>(metadata.hints); + this.ignored = new LinkedHashMap<>(metadata.ignored); } /** @@ -73,6 +79,32 @@ public void add(ItemHint itemHint) { add(this.hints, itemHint.getName(), itemHint, false); } + /** + * Add item ignore. + * @param itemIgnore the item ignore to add + * @since 3.5.0 + */ + public void add(ItemIgnore itemIgnore) { + add(this.ignored, itemIgnore.getName(), itemIgnore, false); + } + + /** + * Remove item meta-data for the given item type and name. + * @param itemType the item type + * @param name the name + * @since 3.5.0 + */ + public void removeMetadata(ItemType itemType, String name) { + List metadata = this.items.get(name); + if (metadata == null) { + return; + } + metadata.removeIf((item) -> item.isOfItemType(itemType)); + if (metadata.isEmpty()) { + this.items.remove(name); + } + } + /** * Merge the content from another {@link ConfigurationMetadata}. * @param metadata the {@link ConfigurationMetadata} instance to merge @@ -84,6 +116,9 @@ public void merge(ConfigurationMetadata metadata) { for (ItemHint itemHint : metadata.getHints()) { add(itemHint); } + for (ItemIgnore itemIgnore : metadata.getIgnored()) { + add(itemIgnore); + } } /** @@ -102,6 +137,14 @@ public List getHints() { return flattenValues(this.hints); } + /** + * Return ignore meta-data. + * @return the ignores + */ + public List getIgnored() { + return flattenValues(this.ignored); + } + protected void mergeItemMetadata(ItemMetadata metadata) { ItemMetadata matching = findMatchingItemMetadata(metadata); if (matching != null) { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/ItemIgnore.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/ItemIgnore.java new file mode 100644 index 000000000000..ec5e7971f326 --- /dev/null +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/ItemIgnore.java @@ -0,0 +1,87 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.configurationprocessor.metadata; + +import java.util.Objects; + +import org.springframework.boot.configurationprocessor.metadata.ItemMetadata.ItemType; + +/** + * Ignored item. + * + * @author Moritz Halbritter + * @since 3.5.0 + */ +public final class ItemIgnore implements Comparable { + + private final ItemType type; + + private final String name; + + private ItemIgnore(ItemType type, String name) { + if (type == null) { + throw new IllegalArgumentException("'type' must not be null"); + } + if (name == null) { + throw new IllegalArgumentException("'name' must not be null"); + } + this.type = type; + this.name = name; + } + + public String getName() { + return this.name; + } + + public ItemType getType() { + return this.type; + } + + @Override + public int compareTo(ItemIgnore other) { + return getName().compareTo(other.getName()); + } + + /** + * Create an ignore for a property with the given name. + * @param name the name + * @return the item ignore + */ + public static ItemIgnore forProperty(String name) { + return new ItemIgnore(ItemType.PROPERTY, name); + } + + @Override + public boolean equals(Object o) { + if (o == null || getClass() != o.getClass()) { + return false; + } + ItemIgnore that = (ItemIgnore) o; + return this.type == that.type && Objects.equals(this.name, that.name); + } + + @Override + public int hashCode() { + return Objects.hash(this.type, this.name); + } + + @Override + public String toString() { + return "ItemIgnore{" + "type=" + this.type + ", name='" + this.name + '\'' + '}'; + } + +} diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/JsonConverter.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/JsonConverter.java index bd6239e19df1..c33cf67f142c 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/JsonConverter.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/JsonConverter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,6 +31,7 @@ * * @author Stephane Nicoll * @author Phillip Webb + * @author Moritz Halbritter */ class JsonConverter { @@ -59,6 +60,21 @@ JSONArray toJsonArray(Collection hints) throws Exception { return jsonArray; } + JSONObject toJsonObject(Collection ignored) throws Exception { + JSONObject result = new JSONObject(); + result.put("properties", ignoreToJsonArray( + ignored.stream().filter((itemIgnore) -> itemIgnore.getType() == ItemType.PROPERTY).toList())); + return result; + } + + private JSONArray ignoreToJsonArray(Collection ignored) throws Exception { + JSONArray result = new JSONArray(); + for (ItemIgnore itemIgnore : ignored) { + result.put(toJsonObject(itemIgnore)); + } + return result; + } + JSONObject toJsonObject(ItemMetadata item) throws Exception { JSONObject jsonObject = new JSONObject(); jsonObject.put("name", item.getName()); @@ -103,6 +119,12 @@ private JSONObject toJsonObject(ItemHint hint) throws Exception { return jsonObject; } + private JSONObject toJsonObject(ItemIgnore ignore) throws Exception { + JSONObject jsonObject = new JSONObject(); + jsonObject.put("name", ignore.getName()); + return jsonObject; + } + private JSONArray getItemHintValues(ItemHint hint) throws Exception { JSONArray values = new JSONArray(); for (ItemHint.ValueHint value : hint.getValues()) { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/JsonMarshaller.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/JsonMarshaller.java index e7d6e84e8a26..e6981d73b6e3 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/JsonMarshaller.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/main/java/org/springframework/boot/configurationprocessor/metadata/JsonMarshaller.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,6 +30,7 @@ import java.util.TreeSet; import org.springframework.boot.configurationprocessor.json.JSONArray; +import org.springframework.boot.configurationprocessor.json.JSONException; import org.springframework.boot.configurationprocessor.json.JSONObject; import org.springframework.boot.configurationprocessor.metadata.ItemMetadata.ItemType; @@ -50,6 +51,7 @@ public void write(ConfigurationMetadata metadata, OutputStream outputStream) thr object.put("groups", converter.toJsonArray(metadata, ItemType.GROUP)); object.put("properties", converter.toJsonArray(metadata, ItemType.PROPERTY)); object.put("hints", converter.toJsonArray(metadata.getHints())); + object.put("ignored", converter.toJsonObject(metadata.getIgnored())); outputStream.write(object.toString(2).getBytes(StandardCharsets.UTF_8)); } catch (Exception ex) { @@ -67,7 +69,7 @@ public ConfigurationMetadata read(InputStream inputStream) throws Exception { ConfigurationMetadata metadata = new ConfigurationMetadata(); JSONObject object = new JSONObject(toString(inputStream)); JsonPath path = JsonPath.root(); - checkAllowedKeys(object, path, "groups", "properties", "hints"); + checkAllowedKeys(object, path, "groups", "properties", "hints", "ignored"); JSONArray groups = object.optJSONArray("groups"); if (groups != null) { for (int i = 0; i < groups.length(); i++) { @@ -88,9 +90,28 @@ public ConfigurationMetadata read(InputStream inputStream) throws Exception { metadata.add(toItemHint((JSONObject) hints.get(i), path.resolve("hints").index(i))); } } + JSONObject ignored = object.optJSONObject("ignored"); + if (ignored != null) { + JsonPath ignoredPath = path.resolve("ignored"); + checkAllowedKeys(ignored, ignoredPath, "properties"); + addIgnoredProperties(metadata, ignored, ignoredPath); + } return metadata; } + private void addIgnoredProperties(ConfigurationMetadata metadata, JSONObject ignored, JsonPath path) + throws JSONException { + JSONArray properties = ignored.optJSONArray("properties"); + if (properties == null) { + return; + } + for (int i = 0; i < properties.length(); i++) { + JSONObject jsonObject = properties.getJSONObject(i); + checkAllowedKeys(jsonObject, path.resolve("properties").index(i), "name"); + metadata.add(ItemIgnore.forProperty(jsonObject.getString("name"))); + } + } + private ItemMetadata toItemMetadata(JSONObject object, JsonPath path, ItemType itemType) throws Exception { switch (itemType) { case GROUP -> checkAllowedKeys(object, path, "name", "type", "description", "sourceType", "sourceMethod"); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationprocessor/ConfigurationMetadataAnnotationProcessorTests.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationprocessor/ConfigurationMetadataAnnotationProcessorTests.java index ac87c915d851..d659e13de759 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationprocessor/ConfigurationMetadataAnnotationProcessorTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationprocessor/ConfigurationMetadataAnnotationProcessorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,6 +22,7 @@ import org.junit.jupiter.api.Test; import org.springframework.boot.configurationprocessor.metadata.ConfigurationMetadata; +import org.springframework.boot.configurationprocessor.metadata.ItemIgnore; import org.springframework.boot.configurationprocessor.metadata.ItemMetadata; import org.springframework.boot.configurationprocessor.metadata.Metadata; import org.springframework.boot.configurationsample.deprecation.Dbcp2Configuration; @@ -38,6 +39,7 @@ import org.springframework.boot.configurationsample.simple.HierarchicalProperties; import org.springframework.boot.configurationsample.simple.HierarchicalPropertiesGrandparent; import org.springframework.boot.configurationsample.simple.HierarchicalPropertiesParent; +import org.springframework.boot.configurationsample.simple.IgnoredProperties; import org.springframework.boot.configurationsample.simple.InnerClassWithPrivateConstructor; import org.springframework.boot.configurationsample.simple.NotAnnotated; import org.springframework.boot.configurationsample.simple.SimpleArrayProperties; @@ -570,4 +572,24 @@ void recordPropertiesWithDescriptions() { .withDescription("last description in Javadoc")); } + @Test + void shouldIgnoreProperties() { + String additionalMetadata = """ + { + "ignored": { + "properties": [ + { + "name": "ignored.prop3" + } + ] + } + } + """; + ConfigurationMetadata metadata = compile(additionalMetadata, IgnoredProperties.class); + assertThat(metadata).has(Metadata.withProperty("ignored.prop1", String.class)); + assertThat(metadata).has(Metadata.withProperty("ignored.prop2", String.class)); + assertThat(metadata).doesNotHave(Metadata.withProperty("ignored.prop3", String.class)); + assertThat(metadata.getIgnored()).containsExactly(ItemIgnore.forProperty("ignored.prop3")); + } + } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationprocessor/metadata/JsonMarshallerTests.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationprocessor/metadata/JsonMarshallerTests.java index 00252fc252e4..e8e0684aea3e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationprocessor/metadata/JsonMarshallerTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationprocessor/metadata/JsonMarshallerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,6 +34,7 @@ * * @author Phillip Webb * @author Stephane Nicoll + * @author Moritz Halbritter */ class JsonMarshallerTests { @@ -174,14 +175,36 @@ void orderingForSamePropertyNamesWithNullSourceType() throws IOException { "\"java.lang.Boolean\"", "\"com.example.bravo.aaa\"", "\"java.lang.Integer\"", "\"com.example.Bar"); } + @Test + void shouldReadIgnoredProperties() throws Exception { + String json = """ + { + "ignored": { + "properties": [ + { + "name": "prop1" + }, + { + "name": "prop2" + } + ] + } + } + """; + ConfigurationMetadata metadata = read(json); + assertThat(metadata.getIgnored()).containsExactly(ItemIgnore.forProperty("prop1"), + ItemIgnore.forProperty("prop2")); + } + @Test void shouldCheckRootFields() { String json = """ { - "groups": [], "properties": [], "hints": [], "dummy": [] + "groups": [], "properties": [], "hints": [], "ignored": {}, "dummy": [] }"""; assertThatException().isThrownBy(() -> read(json)) - .withMessage("Expected only keys [groups, hints, properties], but found additional keys [dummy]. Path: ."); + .withMessage( + "Expected only keys [groups, hints, ignored, properties], but found additional keys [dummy]. Path: ."); } @Test @@ -324,9 +347,41 @@ void shouldCheckHintProviderFields() { "Expected only keys [name, parameters], but found additional keys [dummy]. Path: .hints.[0].providers.[0]"); } - private void read(String json) throws Exception { + @Test + void shouldCheckIgnoredFields() { + String json = """ + { + "ignored": { + "properties": [], + "dummy": {} + } + } + """; + assertThatException().isThrownBy(() -> read(json)) + .withMessage("Expected only keys [properties], but found additional keys [dummy]. Path: .ignored"); + } + + @Test + void shouldCheckIgnoredPropertiesFields() { + String json = """ + { + "ignored": { + "properties": [ + { + "name": "prop1", + "dummy": true + } + ] + } + } + """; + assertThatException().isThrownBy(() -> read(json)) + .withMessage("Expected only keys [name], but found additional keys [dummy]. Path: .ignored.properties.[0]"); + } + + private ConfigurationMetadata read(String json) throws Exception { JsonMarshaller marshaller = new JsonMarshaller(); - marshaller.read(new ByteArrayInputStream(json.getBytes(StandardCharsets.UTF_8))); + return marshaller.read(new ByteArrayInputStream(json.getBytes(StandardCharsets.UTF_8))); } } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/deprecation/Dbcp2Configuration.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/deprecation/Dbcp2Configuration.java index 556fd05531a4..b5c19caa0ff1 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/deprecation/Dbcp2Configuration.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/deprecation/Dbcp2Configuration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ */ public class Dbcp2Configuration { - @ConfigurationProperties(prefix = "spring.datasource.dbcp2") + @ConfigurationProperties("spring.datasource.dbcp2") BasicDataSource basicDataSource() { return new BasicDataSource(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/inheritance/ChildPropertiesConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/inheritance/ChildPropertiesConfig.java index 3ff388df6e13..774dd87f0710 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/inheritance/ChildPropertiesConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/inheritance/ChildPropertiesConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,7 +20,7 @@ public class ChildPropertiesConfig { - @ConfigurationProperties(prefix = "inheritance") + @ConfigurationProperties("inheritance") public ChildProperties childConfig() { return new ChildProperties(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/inheritance/OverrideChildPropertiesConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/inheritance/OverrideChildPropertiesConfig.java index 324d12c2a4dc..7eed108ef439 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/inheritance/OverrideChildPropertiesConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/inheritance/OverrideChildPropertiesConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,7 +20,7 @@ public class OverrideChildPropertiesConfig { - @ConfigurationProperties(prefix = "inheritance") + @ConfigurationProperties("inheritance") public OverrideChildProperties overrideChildProperties() { return new OverrideChildProperties(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteDataProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteDataProperties.java index 51a5251b9a66..4373edd7071a 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteDataProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteDataProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,7 +30,7 @@ * @author Jonas Keßler */ @Data -@ConfigurationProperties(prefix = "accesslevel.overwrite.data") +@ConfigurationProperties("accesslevel.overwrite.data") @SuppressWarnings("unused") public class LombokAccessLevelOverwriteDataProperties { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteDefaultProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteDefaultProperties.java index f8c9d235620c..8f9f3904c05b 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteDefaultProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteDefaultProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,7 +30,7 @@ */ @Getter @Setter -@ConfigurationProperties(prefix = "accesslevel.overwrite.default") +@ConfigurationProperties("accesslevel.overwrite.default") public class LombokAccessLevelOverwriteDefaultProperties { @SuppressWarnings("unused") diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteExplicitProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteExplicitProperties.java index 5acb43ed5718..462ed673e25c 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteExplicitProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelOverwriteExplicitProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,7 +30,7 @@ */ @Getter(AccessLevel.PUBLIC) @Setter(AccessLevel.PUBLIC) -@ConfigurationProperties(prefix = "accesslevel.overwrite.explicit") +@ConfigurationProperties("accesslevel.overwrite.explicit") @SuppressWarnings("unused") public class LombokAccessLevelOverwriteExplicitProperties { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelProperties.java index ebf13fa4d7bc..a5ce2a949298 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokAccessLevelProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ * * @author Jonas Keßler */ -@ConfigurationProperties(prefix = "accesslevel") +@ConfigurationProperties("accesslevel") public class LombokAccessLevelProperties { @Getter(AccessLevel.PUBLIC) diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokDeprecatedProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokDeprecatedProperties.java index d621d716d565..f7f1685475cb 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokDeprecatedProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokDeprecatedProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ */ @Getter @Setter -@ConfigurationProperties(prefix = "deprecated") +@ConfigurationProperties("deprecated") @Deprecated @SuppressWarnings("unused") public class LombokDeprecatedProperties { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokExplicitProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokExplicitProperties.java index 21db60670d6b..54a8d9fb577d 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokExplicitProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokExplicitProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "explicit") +@ConfigurationProperties("explicit") public class LombokExplicitProperties { @Getter diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokInnerClassProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokInnerClassProperties.java index 551ba6ab5872..d346c21f1512 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokInnerClassProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokInnerClassProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ * @author Stephane Nicoll */ @Data -@ConfigurationProperties(prefix = "config") +@ConfigurationProperties("config") @SuppressWarnings("unused") public class LombokInnerClassProperties { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokInnerClassWithGetterProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokInnerClassWithGetterProperties.java index f494646886e0..7143bd64f67a 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokInnerClassWithGetterProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokInnerClassWithGetterProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -21,7 +21,7 @@ import org.springframework.boot.configurationsample.ConfigurationProperties; @Data -@ConfigurationProperties(prefix = "config") +@ConfigurationProperties("config") @SuppressWarnings("unused") public class LombokInnerClassWithGetterProperties { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleDataProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleDataProperties.java index 574e08c6a6c3..de07ab1b30bb 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleDataProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleDataProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Stephane Nicoll */ @Data -@ConfigurationProperties(prefix = "data") +@ConfigurationProperties("data") @SuppressWarnings("unused") public class LombokSimpleDataProperties { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleProperties.java index 64a0e3799a87..84a135b4c1b0 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,7 @@ */ @Getter @Setter -@ConfigurationProperties(prefix = "simple") +@ConfigurationProperties("simple") @SuppressWarnings("unused") public class LombokSimpleProperties { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleValueProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleValueProperties.java index 022e5b413e43..19776e8f0f42 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleValueProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/lombok/LombokSimpleValueProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * @author Mark Jeffrey */ @Value -@ConfigurationProperties(prefix = "value") +@ConfigurationProperties("value") @SuppressWarnings("unused") public class LombokSimpleValueProperties { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/DeprecatedClassMethodConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/DeprecatedClassMethodConfig.java index 7a32353563b0..17c585258f7e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/DeprecatedClassMethodConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/DeprecatedClassMethodConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ @Deprecated public class DeprecatedClassMethodConfig { - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") public Foo foo() { return new Foo(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/DeprecatedMethodConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/DeprecatedMethodConfig.java index e8e3de581efb..dd67de6c7fbc 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/DeprecatedMethodConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/DeprecatedMethodConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ */ public class DeprecatedMethodConfig { - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") @Deprecated public Foo foo() { return new Foo(); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/InvalidMethodConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/InvalidMethodConfig.java index 7b9e106cec81..a45a3315f9fb 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/InvalidMethodConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/InvalidMethodConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,7 +23,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "something") +@ConfigurationProperties("something") public class InvalidMethodConfig { private String name; @@ -36,7 +36,7 @@ public void setName(String name) { this.name = name; } - @ConfigurationProperties(prefix = "invalid") + @ConfigurationProperties("invalid") InvalidMethodConfig foo() { return new InvalidMethodConfig(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/MethodAndClassConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/MethodAndClassConfig.java index ca028eb7a7a7..542e3b5fd838 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/MethodAndClassConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/MethodAndClassConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,7 +36,7 @@ public void setValue(String value) { this.value = value; } - @ConfigurationProperties(prefix = "conflict") + @ConfigurationProperties("conflict") public Foo foo() { return new Foo(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PackagePrivateMethodConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PackagePrivateMethodConfig.java index b1dbd565c09a..7bbf0ffa5217 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PackagePrivateMethodConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PackagePrivateMethodConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ */ public class PackagePrivateMethodConfig { - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") Foo foo() { return new Foo(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PrivateMethodConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PrivateMethodConfig.java index 44f033bf8091..af666aac8a6e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PrivateMethodConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PrivateMethodConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ */ public class PrivateMethodConfig { - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") private Foo foo() { return new Foo(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/ProtectedMethodConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/ProtectedMethodConfig.java index 6d5f3c409002..43c20d05ccd4 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/ProtectedMethodConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/ProtectedMethodConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ */ public class ProtectedMethodConfig { - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") protected Foo foo() { return new Foo(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PublicMethodConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PublicMethodConfig.java index 677ffbb33561..3118849afdde 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PublicMethodConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/PublicMethodConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ */ public class PublicMethodConfig { - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") public Foo foo() { return new Foo(); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/SingleConstructorMethodConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/SingleConstructorMethodConfig.java index 7bea9f880b30..13d3e546e66a 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/SingleConstructorMethodConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/method/SingleConstructorMethodConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -27,7 +27,7 @@ @SuppressWarnings("unused") public class SingleConstructorMethodConfig { - @ConfigurationProperties(prefix = "foo") + @ConfigurationProperties("foo") public Foo foo() { return new Foo(new Object()); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/ClassWithNestedProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/ClassWithNestedProperties.java index ed9ab56bdf2a..7022cc48295b 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/ClassWithNestedProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/ClassWithNestedProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,7 @@ public void setParentClassProperty(int parentClassProperty) { } - @ConfigurationProperties(prefix = "nestedChildProps") + @ConfigurationProperties("nestedChildProps") public static class NestedChildClass extends NestedParentClass { private int childClassProperty = 20; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/DeprecatedProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/DeprecatedProperties.java index d29acb0a348c..83be6a84c68c 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/DeprecatedProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/DeprecatedProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ * @deprecated deprecated */ @Deprecated -@ConfigurationProperties(prefix = "deprecated") +@ConfigurationProperties("deprecated") public class DeprecatedProperties { private String name; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/HierarchicalProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/HierarchicalProperties.java index cbd0ff27be98..76e99e52abf1 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/HierarchicalProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/HierarchicalProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,7 +23,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "hierarchical") +@ConfigurationProperties("hierarchical") public class HierarchicalProperties extends HierarchicalPropertiesParent { private String third = "three"; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/IgnoredProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/IgnoredProperties.java new file mode 100644 index 000000000000..a68138c03961 --- /dev/null +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/IgnoredProperties.java @@ -0,0 +1,59 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.configurationsample.simple; + +import org.springframework.boot.configurationsample.ConfigurationProperties; + +/** + * Configuration properties where some of them are being ignored. + * + * @author Moritz Halbritter + */ +@ConfigurationProperties("ignored") +public class IgnoredProperties { + + private String prop1; + + private String prop2; + + private String prop3; + + public String getProp1() { + return this.prop1; + } + + public void setProp1(String prop1) { + this.prop1 = prop1; + } + + public String getProp2() { + return this.prop2; + } + + public void setProp2(String prop2) { + this.prop2 = prop2; + } + + public String getProp3() { + return this.prop3; + } + + public void setProp3(String prop3) { + this.prop3 = prop3; + } + +} diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/InnerClassWithPrivateConstructor.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/InnerClassWithPrivateConstructor.java index 474d7691b63d..76e35f9a6243 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/InnerClassWithPrivateConstructor.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/InnerClassWithPrivateConstructor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,7 +23,7 @@ * * @author Phillip Webb */ -@ConfigurationProperties(prefix = "config") +@ConfigurationProperties("config") public class InnerClassWithPrivateConstructor { private Nested nested = new Nested("whatever"); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleCollectionProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleCollectionProperties.java index 019ba7f956b3..672b1ad90fe2 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleCollectionProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleCollectionProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,7 +30,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "collection") +@ConfigurationProperties("collection") public class SimpleCollectionProperties { private Map integersToNames; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleProperties.java index 43d672796074..620cf0b6cc1f 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,7 +26,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "simple") +@ConfigurationProperties("simple") public class SimpleProperties { /** diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleTypeProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleTypeProperties.java index ef9ab15d8021..56b4856d477a 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleTypeProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/simple/SimpleTypeProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,7 +23,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "simple.type") +@ConfigurationProperties("simple.type") public class SimpleTypeProperties { private String myString; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/BuilderPojo.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/BuilderPojo.java index b42aa524a0fd..c0c0902fa463 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/BuilderPojo.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/BuilderPojo.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,7 +23,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "builder") +@ConfigurationProperties("builder") public class BuilderPojo { private String name; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/ExcludedTypesPojo.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/ExcludedTypesPojo.java index 2942c9287bca..fc7cd443fcc4 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/ExcludedTypesPojo.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/ExcludedTypesPojo.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,7 +29,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "excluded") +@ConfigurationProperties("excluded") public class ExcludedTypesPojo { private String name; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassHierarchicalProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassHierarchicalProperties.java index ab1c61ef8394..1100f7ddbbb1 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassHierarchicalProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassHierarchicalProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ * * @author Madhura Bhave */ -@ConfigurationProperties(prefix = "config") +@ConfigurationProperties("config") public class InnerClassHierarchicalProperties { private Foo foo; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassProperties.java index 049371b45fd7..16e3d942d58a 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -24,7 +24,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "config") +@ConfigurationProperties("config") public class InnerClassProperties { private final Foo first = new Foo(); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassRootConfig.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassRootConfig.java index d8cf493bd307..60eda5a355d4 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassRootConfig.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InnerClassRootConfig.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,7 @@ */ public class InnerClassRootConfig { - @ConfigurationProperties(prefix = "config") + @ConfigurationProperties("config") public static class Config { private String name; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InvalidAccessorProperties.java b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InvalidAccessorProperties.java index de7cdf2a82f3..ae85e37e4acd 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InvalidAccessorProperties.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-configuration-processor/src/test/java/org/springframework/boot/configurationsample/specific/InvalidAccessorProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,7 +23,7 @@ * * @author Stephane Nicoll */ -@ConfigurationProperties(prefix = "config") +@ConfigurationProperties("config") public class InvalidAccessorProperties { private String name; diff --git a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/dockerTest/java/org/springframework/boot/gradle/tasks/bundling/BootBuildImageIntegrationTests.java b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/dockerTest/java/org/springframework/boot/gradle/tasks/bundling/BootBuildImageIntegrationTests.java index cad3e53ba22e..aba95aabd28c 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/dockerTest/java/org/springframework/boot/gradle/tasks/bundling/BootBuildImageIntegrationTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/dockerTest/java/org/springframework/boot/gradle/tasks/bundling/BootBuildImageIntegrationTests.java @@ -482,7 +482,7 @@ void failsWithInvalidImageName() throws IOException { writeLongNameResource(); BuildResult result = this.gradleBuild.buildAndFail("bootBuildImage", "--imageName=example/Invalid-Image-Name"); assertThat(result.task(":bootBuildImage").getOutcome()).isEqualTo(TaskOutcome.FAILED); - assertThat(result.getOutput()).containsPattern("Unable to parse image reference") + assertThat(result.getOutput()).containsPattern("must be an image reference") .containsPattern("example/Invalid-Image-Name"); } @@ -501,7 +501,7 @@ void failsWithInvalidTag() throws IOException { writeLongNameResource(); BuildResult result = this.gradleBuild.buildAndFail("bootBuildImage"); assertThat(result.task(":bootBuildImage").getOutcome()).isEqualTo(TaskOutcome.FAILED); - assertThat(result.getOutput()).containsPattern("Unable to parse image reference") + assertThat(result.getOutput()).containsPattern("must be an image reference") .containsPattern("example/Invalid-Tag-Name"); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/BootJar.java b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/BootJar.java index 27454860a172..affd7f35cac3 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/BootJar.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/BootJar.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -154,10 +154,8 @@ protected CopyAction createCopyAction() { jarmodeToolsLocation); } - @SuppressWarnings("removal") private boolean isIncludeJarmodeTools() { - return Boolean.TRUE.equals(this.getIncludeTools().get()) - && Boolean.TRUE.equals(this.layered.getIncludeLayerTools().get()); + return Boolean.TRUE.equals(this.getIncludeTools().get()); } @Override diff --git a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/BootWar.java b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/BootWar.java index 2dfaee6ffe04..5d1ba638f501 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/BootWar.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/BootWar.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -128,10 +128,8 @@ protected CopyAction createCopyAction() { layerResolver, jarmodeToolsLocation); } - @SuppressWarnings("removal") private boolean isIncludeJarmodeTools() { - return Boolean.TRUE.equals(this.getIncludeTools().get()) - && Boolean.TRUE.equals(this.layered.getIncludeLayerTools().get()); + return Boolean.TRUE.equals(this.getIncludeTools().get()); } @Override diff --git a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/LayeredSpec.java b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/LayeredSpec.java index 98898783a26e..afaa323a4f83 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/LayeredSpec.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/main/java/org/springframework/boot/gradle/tasks/bundling/LayeredSpec.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -64,20 +64,8 @@ public LayeredSpec(ObjectFactory objects) { this.application = objects.newInstance(ApplicationSpec.class); this.dependencies = objects.newInstance(DependenciesSpec.class); getEnabled().convention(true); - getIncludeLayerTools().convention(true); } - /** - * Returns whether the layer tools should be included as a dependency in the layered - * archive. - * @return whether the layer tools should be included - * @since 3.0.0 - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of {@code includeTools}. - */ - @Input - @Deprecated(since = "3.3.0", forRemoval = true) - public abstract Property getIncludeLayerTools(); - /** * Returns whether the layers.idx should be included in the archive. * @return whether the layers.idx should be included diff --git a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/java/org/springframework/boot/gradle/tasks/bundling/AbstractBootArchiveIntegrationTests.java b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/java/org/springframework/boot/gradle/tasks/bundling/AbstractBootArchiveIntegrationTests.java index df29ef1ef0f9..4fb7f0ebf4e5 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/java/org/springframework/boot/gradle/tasks/bundling/AbstractBootArchiveIntegrationTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/java/org/springframework/boot/gradle/tasks/bundling/AbstractBootArchiveIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -321,18 +321,6 @@ void notUpToDateWhenBuiltWithoutLayersAndThenWithLayers() { .getOutcome()).isEqualTo(TaskOutcome.SUCCESS); } - @TestTemplate - void notUpToDateWhenBuiltWithLayerToolsAndThenWithoutLayerTools() { - assertThat(this.gradleBuild.scriptProperty("layerTools", "") - .build(this.taskName) - .task(":" + this.taskName) - .getOutcome()).isEqualTo(TaskOutcome.SUCCESS); - assertThat(this.gradleBuild.scriptProperty("layerTools", "includeLayerTools = false") - .build(this.taskName) - .task(":" + this.taskName) - .getOutcome()).isEqualTo(TaskOutcome.SUCCESS); - } - @TestTemplate void notUpToDateWhenBuiltWithToolsAndThenWithoutTools() { assertThat(this.gradleBuild.scriptProperty("includeTools", "") diff --git a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/java/org/springframework/boot/gradle/tasks/bundling/AbstractBootArchiveTests.java b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/java/org/springframework/boot/gradle/tasks/bundling/AbstractBootArchiveTests.java index 42168e1a9b60..2c9c2213b5b9 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/java/org/springframework/boot/gradle/tasks/bundling/AbstractBootArchiveTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/java/org/springframework/boot/gradle/tasks/bundling/AbstractBootArchiveTests.java @@ -626,14 +626,6 @@ void shouldAddToolsToTheJar() throws IOException { assertThat(entryNames).isNotEmpty().contains(this.libPath + JarModeLibrary.TOOLS.getName()); } - @Test - @SuppressWarnings("removal") - void whenArchiveIsLayeredAndIncludeLayerToolsIsFalseThenLayerToolsAreNotAddedToTheJar() throws IOException { - List entryNames = getEntryNames( - createLayeredJar((configuration) -> configuration.getIncludeLayerTools().set(false))); - assertThat(entryNames).isNotEmpty().doesNotContain(this.libPath + JarModeLibrary.TOOLS.getName()); - } - @Test void whenIncludeToolsIsFalseThenToolsAreNotAddedToTheJar() throws IOException { this.task.getIncludeTools().set(false); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/resources/org/springframework/boot/gradle/tasks/bundling/BootJarIntegrationTests-notUpToDateWhenBuiltWithLayerToolsAndThenWithoutLayerTools.gradle b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/resources/org/springframework/boot/gradle/tasks/bundling/BootJarIntegrationTests-notUpToDateWhenBuiltWithLayerToolsAndThenWithoutLayerTools.gradle deleted file mode 100644 index 7530a605b9e5..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/resources/org/springframework/boot/gradle/tasks/bundling/BootJarIntegrationTests-notUpToDateWhenBuiltWithLayerToolsAndThenWithoutLayerTools.gradle +++ /dev/null @@ -1,11 +0,0 @@ -plugins { - id 'java' - id 'org.springframework.boot' version '{version}' -} - -bootJar { - mainClass = 'com.example.Application' - layered { - {layerTools} - } -} diff --git a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/resources/org/springframework/boot/gradle/tasks/bundling/BootWarIntegrationTests-notUpToDateWhenBuiltWithLayerToolsAndThenWithoutLayerTools.gradle b/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/resources/org/springframework/boot/gradle/tasks/bundling/BootWarIntegrationTests-notUpToDateWhenBuiltWithLayerToolsAndThenWithoutLayerTools.gradle deleted file mode 100644 index aa8d6fa822f6..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-gradle-plugin/src/test/resources/org/springframework/boot/gradle/tasks/bundling/BootWarIntegrationTests-notUpToDateWhenBuiltWithLayerToolsAndThenWithoutLayerTools.gradle +++ /dev/null @@ -1,12 +0,0 @@ -plugins { - id 'java' - id 'org.springframework.boot' version '{version}' - id 'war' -} - -bootWar { - mainClass = 'com.example.Application' - layered { - {layerTools} - } -} diff --git a/spring-boot-project/spring-boot-tools/spring-boot-jarmode-tools/src/main/java/org/springframework/boot/jarmode/tools/IndexedJarStructure.java b/spring-boot-project/spring-boot-tools/spring-boot-jarmode-tools/src/main/java/org/springframework/boot/jarmode/tools/IndexedJarStructure.java index 855160ba1c62..a50c18ec3336 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-jarmode-tools/src/main/java/org/springframework/boot/jarmode/tools/IndexedJarStructure.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-jarmode-tools/src/main/java/org/springframework/boot/jarmode/tools/IndexedJarStructure.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -128,13 +128,13 @@ public Manifest createLauncherManifest(UnaryOperator libraryTransformer) } private String toStructureDependency(String libEntryName) { - Assert.state(libEntryName.startsWith(this.libLocation), "Invalid library location " + libEntryName); + Assert.state(libEntryName.startsWith(this.libLocation), () -> "Invalid library location " + libEntryName); return libEntryName.substring(this.libLocation.length()); } private static String getMandatoryAttribute(Manifest manifest, String attribute) { String value = manifest.getMainAttributes().getValue(attribute); - Assert.state(value != null, "Manifest attribute '" + attribute + "' is mandatory"); + Assert.state(value != null, () -> "Manifest attribute '" + attribute + "' is mandatory"); return value; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-jarmode-tools/src/main/java/org/springframework/boot/jarmode/tools/IndexedLayers.java b/spring-boot-project/spring-boot-tools/spring-boot-jarmode-tools/src/main/java/org/springframework/boot/jarmode/tools/IndexedLayers.java index 0469aeed8363..2de7e22fce52 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-jarmode-tools/src/main/java/org/springframework/boot/jarmode/tools/IndexedLayers.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-jarmode-tools/src/main/java/org/springframework/boot/jarmode/tools/IndexedLayers.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -60,7 +60,7 @@ class IndexedLayers implements Layers { this.layers.put(line.substring(3, line.length() - 2), contents); } else if (line.startsWith(" - ")) { - Assert.notNull(contents, "Contents must not be null. Check if the index file is malformed!"); + Assert.state(contents != null, "Contents must not be null. Check if the index file is malformed!"); contents.add(line.substring(5, line.length() - 1)); } else { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/Layer.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/Layer.java index 544a7ec772c3..1141ec942a68 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/Layer.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/Layer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,7 +40,7 @@ public class Layer { * @param name the name of the layer. */ public Layer(String name) { - Assert.hasText(name, "Name must not be empty"); + Assert.hasText(name, "'name' must not be empty"); Assert.isTrue(PATTERN.matcher(name).matches(), () -> "Malformed layer name '" + name + "'"); Assert.isTrue(!name.equalsIgnoreCase("ext") && !name.toLowerCase(Locale.ROOT).startsWith("springboot"), () -> "Layer name '" + name + "' is reserved"); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/Packager.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/Packager.java index 07fa54e66352..d75362861b3e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/Packager.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/Packager.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -107,9 +107,9 @@ public abstract class Packager { * @param source the source archive file to package */ protected Packager(File source) { - Assert.notNull(source, "Source file must not be null"); + Assert.notNull(source, "'source' file must not be null"); Assert.isTrue(source.exists() && source.isFile(), - () -> "Source must refer to an existing file, got " + source.getAbsolutePath()); + () -> "'source' must refer to an existing file, got " + source.getAbsolutePath()); this.source = source.getAbsoluteFile(); } @@ -163,7 +163,7 @@ public void setLayoutFactory(LayoutFactory layoutFactory) { * @param layers the jar layers */ public void setLayers(Layers layers) { - Assert.notNull(layers, "Layers must not be null"); + Assert.notNull(layers, "'layers' must not be null"); this.layers = layers; this.layersIndex = new LayersIndex(layers); } @@ -204,7 +204,7 @@ protected final void write(JarFile sourceJar, Libraries libraries, AbstractJarWr protected final void write(JarFile sourceJar, Libraries libraries, AbstractJarWriter writer, boolean ensureReproducibleBuild) throws IOException { - Assert.notNull(libraries, "Libraries must not be null"); + Assert.notNull(libraries, "'libraries' must not be null"); write(sourceJar, writer, new PackagedLibraries(libraries, ensureReproducibleBuild)); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/ApplicationContentFilter.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/ApplicationContentFilter.java index 734ea7e2605b..68dc8371f648 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/ApplicationContentFilter.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/ApplicationContentFilter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,7 +34,7 @@ public class ApplicationContentFilter implements ContentFilter { private final String pattern; public ApplicationContentFilter(String pattern) { - Assert.hasText(pattern, "Pattern must not be empty"); + Assert.hasText(pattern, "'pattern' must not be empty"); this.pattern = pattern; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/CustomLayers.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/CustomLayers.java index c8334b5b1607..0264d2318c5e 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/CustomLayers.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/CustomLayers.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -44,10 +44,10 @@ public class CustomLayers implements Layers { public CustomLayers(List layers, List> applicationSelectors, List> librarySelectors) { - Assert.notNull(layers, "Layers must not be null"); - Assert.notNull(applicationSelectors, "ApplicationSelectors must not be null"); + Assert.notNull(layers, "'layers' must not be null"); + Assert.notNull(applicationSelectors, "'applicationSelectors' must not be null"); validateSelectorLayers(applicationSelectors, layers); - Assert.notNull(librarySelectors, "LibrarySelectors must not be null"); + Assert.notNull(librarySelectors, "'librarySelectors' must not be null"); validateSelectorLayers(librarySelectors, layers); this.layers = new ArrayList<>(layers); this.applicationSelectors = new ArrayList<>(applicationSelectors); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/IncludeExcludeContentSelector.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/IncludeExcludeContentSelector.java index 947f3959d271..3d7c5ff0fbfc 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/IncludeExcludeContentSelector.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/IncludeExcludeContentSelector.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -47,8 +47,8 @@ public IncludeExcludeContentSelector(Layer layer, List> include public IncludeExcludeContentSelector(Layer layer, List includes, List excludes, Function> filterFactory) { - Assert.notNull(layer, "Layer must not be null"); - Assert.notNull(filterFactory, "FilterFactory must not be null"); + Assert.notNull(layer, "'layer' must not be null"); + Assert.notNull(filterFactory, "'filterFactory' must not be null"); this.layer = layer; this.includes = (includes != null) ? adapt(includes, filterFactory) : Collections.emptyList(); this.excludes = (excludes != null) ? adapt(excludes, filterFactory) : Collections.emptyList(); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/LibraryContentFilter.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/LibraryContentFilter.java index 295994c6bd95..c7fd66334c06 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/LibraryContentFilter.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/main/java/org/springframework/boot/loader/tools/layer/LibraryContentFilter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,7 +36,7 @@ public class LibraryContentFilter implements ContentFilter { private final Pattern pattern; public LibraryContentFilter(String coordinatesPattern) { - Assert.hasText(coordinatesPattern, "CoordinatesPattern must not be empty"); + Assert.hasText(coordinatesPattern, "'coordinatesPattern' must not be empty"); StringBuilder regex = new StringBuilder(); for (int i = 0; i < coordinatesPattern.length(); i++) { char c = coordinatesPattern.charAt(i); diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/AbstractPackagerTests.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/AbstractPackagerTests.java index 8962c11a4d23..6bd099839487 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/AbstractPackagerTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/AbstractPackagerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -182,7 +182,7 @@ void nullLibraries() throws Exception { this.testJarFile.addClass("a/b/C.class", ClassWithMainMethod.class); P packager = createPackager(); assertThatIllegalArgumentException().isThrownBy(() -> execute(packager, null)) - .withMessageContaining("Libraries must not be null"); + .withMessageContaining("'libraries' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/LayerTests.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/LayerTests.java index 0328e2b9f10b..1dd053b350e1 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/LayerTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/LayerTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,12 +31,12 @@ class LayerTests { @Test void createWhenNameIsNullThrowsException() { - assertThatIllegalArgumentException().isThrownBy(() -> new Layer(null)).withMessage("Name must not be empty"); + assertThatIllegalArgumentException().isThrownBy(() -> new Layer(null)).withMessage("'name' must not be empty"); } @Test void createWhenNameIsEmptyThrowsException() { - assertThatIllegalArgumentException().isThrownBy(() -> new Layer("")).withMessage("Name must not be empty"); + assertThatIllegalArgumentException().isThrownBy(() -> new Layer("")).withMessage("'name' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/ApplicationContentFilterTests.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/ApplicationContentFilterTests.java index fd2e7838f998..82ea46b647b1 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/ApplicationContentFilterTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/ApplicationContentFilterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,13 +33,13 @@ class ApplicationContentFilterTests { @Test void createWhenPatternIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new ApplicationContentFilter(null)) - .withMessage("Pattern must not be empty"); + .withMessage("'pattern' must not be empty"); } @Test void createWhenPatternIsEmptyThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new ApplicationContentFilter("")) - .withMessage("Pattern must not be empty"); + .withMessage("'pattern' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/IncludeExcludeContentSelectorTests.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/IncludeExcludeContentSelectorTests.java index f012acdcab32..9a74a64b72cd 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/IncludeExcludeContentSelectorTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/IncludeExcludeContentSelectorTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,14 +42,14 @@ void createWhenLayerIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy( () -> new IncludeExcludeContentSelector<>(null, Collections.emptyList(), Collections.emptyList())) - .withMessage("Layer must not be null"); + .withMessage("'layer' must not be null"); } @Test void createWhenFactoryIsNullThrowsException() { assertThatIllegalArgumentException() .isThrownBy(() -> new IncludeExcludeContentSelector<>(LAYER, null, null, null)) - .withMessage("FilterFactory must not be null"); + .withMessage("'filterFactory' must not be null"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/LibraryContentFilterTests.java b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/LibraryContentFilterTests.java index b6dd41c06cc3..90638af50ec5 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/LibraryContentFilterTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-loader-tools/src/test/java/org/springframework/boot/loader/tools/layer/LibraryContentFilterTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,13 +38,13 @@ class LibraryContentFilterTests { @Test void createWhenCoordinatesPatternIsNullThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new LibraryContentFilter(null)) - .withMessage("CoordinatesPattern must not be empty"); + .withMessage("'coordinatesPattern' must not be empty"); } @Test void createWhenCoordinatesPatternIsEmptyThrowsException() { assertThatIllegalArgumentException().isThrownBy(() -> new LibraryContentFilter("")) - .withMessage("CoordinatesPattern must not be empty"); + .withMessage("'coordinatesPattern' must not be empty"); } @Test diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/build.gradle b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/build.gradle index 83ba59974aaf..21ede5e5b916 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/build.gradle +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/build.gradle @@ -82,7 +82,7 @@ dependencies { mavenRepository(project(path: ":spring-boot-project:spring-boot-docker-compose", configuration: "mavenRepository")) mavenRepository(project(path: ":spring-boot-project:spring-boot-starters:spring-boot-starter-parent", configuration: "mavenRepository")) - versionProperties(project(path: ":spring-boot-project:spring-boot-dependencies", configuration: "effectiveBom")) + versionProperties(project(path: ":spring-boot-project:spring-boot-dependencies", configuration: "resolvedBom")) } ext { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/dockerTest/java/org/springframework/boot/maven/BuildImageTests.java b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/dockerTest/java/org/springframework/boot/maven/BuildImageTests.java index 38abfece174e..7519cd326124 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/dockerTest/java/org/springframework/boot/maven/BuildImageTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/dockerTest/java/org/springframework/boot/maven/BuildImageTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/docs/antora/modules/maven-plugin/examples/aot-test/pom.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/docs/antora/modules/maven-plugin/examples/aot-test/pom.xml deleted file mode 100644 index b20f0d0dcc8b..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/docs/antora/modules/maven-plugin/examples/aot-test/pom.xml +++ /dev/null @@ -1,25 +0,0 @@ - - - 4.0.0 - aot - - - - - org.springframework.boot - spring-boot-maven-plugin - - - process-test-aot - - process-test-aot - - - - - - - - - - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/docs/antora/modules/maven-plugin/pages/aot.adoc b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/docs/antora/modules/maven-plugin/pages/aot.adoc index 442f2877bf78..fcbec79deefe 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/docs/antora/modules/maven-plugin/pages/aot.adoc +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/docs/antora/modules/maven-plugin/pages/aot.adoc @@ -90,17 +90,33 @@ include::partial$goals/process-aot.adoc[leveloffset=+1] == Processing Tests The AOT engine can be applied to JUnit 5 tests that use Spring's Test Context Framework. -Suitable tests are processed by the AOT engine in order to generate `ApplicationContextInitializer` code. +Those tests are processed by the AOT engine and are then executed in a native image. -To configure your application to use this feature, add an execution for the `process-test-aot` goal, as shown in the following example: +Just like <>, the `spring-boot-starter-parent` defines a `nativeTest` profile that can be used to streamline the steps required to execute your tests in a native image. + +The `nativeTest` profile configures the following: + +* Execution of `process-test-aot` when the Spring Boot Maven Plugin is applied on a project. +* Execution of `test` when the {url-native-build-tools-docs-maven-plugin}[Native Build Tools Maven Plugin] is applied on a project. +The execution defines sensible defaults, in particular: +** Making sure the plugin uses the raw classpath, and not the main jar file as it does not understand our repackaged jar format. +** Validate that a suitable GraalVM version is available. +** Download third-party reachability metadata. + +To benefit from the `nativeTest` profile, a module that represents an application should define two plugins, as shown in the following example: [source,xml,indent=0,subs="verbatim,attributes"] ---- -include::example$aot-test/pom.xml[tags=aot] +include::example$aot-native/pom.xml[tags=aot-native] ---- -TIP: If you are using `spring-boot-starter-parent`, this execution is automatically configured if you enable the `nativeTest` profile. +Once the above is in place for each module that needs this feature, you can build your multi-modules project and execute your tests in a native image in the relevant sub-modules, as shown in the following example: + +[source,shell] +---- +$ mvn test -PnativeTest +---- -As with application AOT processing, the `BeanFactory` is fully prepared at build-time. +NOTE: As with application AOT processing, the `BeanFactory` is fully prepared at build-time. include::partial$goals/process-test-aot.adoc[leveloffset=+1] diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/java/org/springframework/boot/maven/JarIntegrationTests.java b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/java/org/springframework/boot/maven/JarIntegrationTests.java index 2bbac630e947..7e9c578c9b0f 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/java/org/springframework/boot/maven/JarIntegrationTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/java/org/springframework/boot/maven/JarIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -408,19 +408,6 @@ void whenJarIsRepackagedWithTheLayersDisabledDoesNotContainLayersIndex(MavenBuil }); } - @TestTemplate - void whenJarIsRepackagedWithTheLayersEnabledAndLayerToolsExcluded(MavenBuild mavenBuild) { - mavenBuild.project("jar-layered-no-layer-tools").execute((project) -> { - File repackaged = new File(project, "jar/target/jar-layered-0.0.1.BUILD-SNAPSHOT.jar"); - assertThat(jar(repackaged)).hasEntryWithNameStartingWith("BOOT-INF/classes/") - .hasEntryWithNameStartingWith("BOOT-INF/lib/jar-release") - .hasEntryWithNameStartingWith("BOOT-INF/lib/jar-snapshot") - .hasEntryWithNameStartingWith("BOOT-INF/layers.idx") - .doesNotHaveEntryWithNameStartingWith( - "BOOT-INF/lib/" + JarModeLibrary.TOOLS.getCoordinates().getArtifactId()); - }); - } - @TestTemplate void whenJarIsRepackagedWithToolsExclude(MavenBuild mavenBuild) { mavenBuild.project("jar-no-tools").execute((project) -> { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/java/org/springframework/boot/maven/WarIntegrationTests.java b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/java/org/springframework/boot/maven/WarIntegrationTests.java index d9038197cbf1..1edb33e51c45 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/java/org/springframework/boot/maven/WarIntegrationTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/java/org/springframework/boot/maven/WarIntegrationTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -184,19 +184,6 @@ void whenWarIsRepackagedWithTheLayersDisabledDoesNotContainLayersIndex(MavenBuil }); } - @TestTemplate - void whenWarIsRepackagedWithTheLayersEnabledAndLayerToolsExcluded(MavenBuild mavenBuild) { - mavenBuild.project("war-layered-no-layer-tools").execute((project) -> { - File repackaged = new File(project, "war/target/war-layered-0.0.1.BUILD-SNAPSHOT.war"); - assertThat(jar(repackaged)).hasEntryWithNameStartingWith("WEB-INF/classes/") - .hasEntryWithNameStartingWith("WEB-INF/lib/jar-release") - .hasEntryWithNameStartingWith("WEB-INF/lib/jar-snapshot") - .hasEntryWithNameStartingWith("WEB-INF/layers.idx") - .doesNotHaveEntryWithNameStartingWith( - "WEB-INF/lib/" + JarModeLibrary.TOOLS.getCoordinates().getArtifactId()); - }); - } - @TestTemplate void whenWarIsRepackagedWithToolsExclude(MavenBuild mavenBuild) { mavenBuild.project("war-no-tools").execute((project) -> { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-custom/jar/src/layers.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-custom/jar/src/layers.xml index cd2a48c9b4cd..ee870e78305c 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-custom/jar/src/layers.xml +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-custom/jar/src/layers.xml @@ -1,7 +1,7 @@ + https://www.springframework.org/schema/layers/layers-3.5.xsd"> **/application*.* diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/jar-release/pom.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/jar-release/pom.xml deleted file mode 100644 index a06fe545f187..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/jar-release/pom.xml +++ /dev/null @@ -1,11 +0,0 @@ - - - 4.0.0 - org.springframework.boot.maven.it - jar-release - 0.0.1.RELEASE - jar - jar - Release Jar dependency - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/jar-snapshot/pom.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/jar-snapshot/pom.xml deleted file mode 100644 index ab31e719baf5..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/jar-snapshot/pom.xml +++ /dev/null @@ -1,11 +0,0 @@ - - - 4.0.0 - org.springframework.boot.maven.it - jar-snapshot - 0.0.1.BUILD-SNAPSHOT - jar - jar - Snapshot Jar dependency - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/jar/pom.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/jar/pom.xml deleted file mode 100644 index b3db8941df92..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/jar/pom.xml +++ /dev/null @@ -1,46 +0,0 @@ - - - 4.0.0 - org.springframework.boot.maven.it - jar-layered - 0.0.1.BUILD-SNAPSHOT - - UTF-8 - @java.version@ - @java.version@ - - - - - @project.groupId@ - @project.artifactId@ - @project.version@ - - - - repackage - - - - false - - - - - - - - - - org.springframework.boot.maven.it - jar-snapshot - 0.0.1.BUILD-SNAPSHOT - - - org.springframework.boot.maven.it - jar-release - 0.0.1.RELEASE - - - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/pom.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/pom.xml deleted file mode 100644 index fdd98953811a..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/jar-layered-no-layer-tools/pom.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - 4.0.0 - org.springframework.boot.maven.it - aggregator - 0.0.1.BUILD-SNAPSHOT - pom - - UTF-8 - @java.version@ - @java.version@ - - - jar-snapshot - jar-release - jar - - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-custom/war/src/layers.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-custom/war/src/layers.xml index cd2a48c9b4cd..ee870e78305c 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-custom/war/src/layers.xml +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-custom/war/src/layers.xml @@ -1,7 +1,7 @@ + https://www.springframework.org/schema/layers/layers-3.5.xsd"> **/application*.* diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/jar-release/pom.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/jar-release/pom.xml deleted file mode 100644 index a06fe545f187..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/jar-release/pom.xml +++ /dev/null @@ -1,11 +0,0 @@ - - - 4.0.0 - org.springframework.boot.maven.it - jar-release - 0.0.1.RELEASE - jar - jar - Release Jar dependency - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/jar-snapshot/pom.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/jar-snapshot/pom.xml deleted file mode 100644 index ab31e719baf5..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/jar-snapshot/pom.xml +++ /dev/null @@ -1,11 +0,0 @@ - - - 4.0.0 - org.springframework.boot.maven.it - jar-snapshot - 0.0.1.BUILD-SNAPSHOT - jar - jar - Snapshot Jar dependency - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/pom.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/pom.xml deleted file mode 100644 index 60503bdf68f8..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/pom.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - 4.0.0 - org.springframework.boot.maven.it - aggregator - 0.0.1.BUILD-SNAPSHOT - pom - - UTF-8 - @java.version@ - @java.version@ - - - jar-snapshot - jar-release - war - - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/war/pom.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/war/pom.xml deleted file mode 100644 index bc367f75f113..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/war/pom.xml +++ /dev/null @@ -1,69 +0,0 @@ - - - 4.0.0 - - org.springframework.boot.maven.it - aggregator - 0.0.1.BUILD-SNAPSHOT - - war-layered - war - war - - - - @project.groupId@ - @project.artifactId@ - @project.version@ - - - - repackage - - - - false - - - - - - - org.apache.maven.plugins - maven-war-plugin - @maven-war-plugin.version@ - - - - Foo - - - - - - - - - org.springframework - spring-context - @spring-framework.version@ - - - jakarta.servlet - jakarta.servlet-api - @jakarta-servlet.version@ - provided - - - org.springframework.boot.maven.it - jar-snapshot - 0.0.1.BUILD-SNAPSHOT - - - org.springframework.boot.maven.it - jar-release - 0.0.1.RELEASE - - - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/war/src/main/webapp/index.html b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/war/src/main/webapp/index.html deleted file mode 100644 index 18ecdcb795c3..000000000000 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/intTest/projects/war-layered-no-layer-tools/war/src/main/webapp/index.html +++ /dev/null @@ -1 +0,0 @@ - diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/AbstractPackagerMojo.java b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/AbstractPackagerMojo.java index 3fbfe48e7108..ffc0848a1e45 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/AbstractPackagerMojo.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/AbstractPackagerMojo.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -180,12 +180,8 @@ protected

P getConfiguredPackager(Supplier

supplier) { return packager; } - @SuppressWarnings("removal") private boolean getIncludeRelevantJarModeJars() { - if (!this.includeTools) { - return false; - } - return this.layers.isIncludeLayerTools(); + return this.includeTools; } private CustomLayers getCustomLayers(File configuration) { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/BuildImageMojo.java b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/BuildImageMojo.java index c5dc26876a58..36b45af6ed07 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/BuildImageMojo.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/BuildImageMojo.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -107,7 +107,7 @@ public abstract class BuildImageMojo extends AbstractPackagerMojo { * property. * @since 2.3.0 */ - @Parameter(property = "spring-boot.build-image.imageName", readonly = true) + @Parameter(property = "spring-boot.build-image.imageName") String imageName; /** @@ -115,14 +115,14 @@ public abstract class BuildImageMojo extends AbstractPackagerMojo { * property. * @since 2.3.0 */ - @Parameter(property = "spring-boot.build-image.builder", readonly = true) + @Parameter(property = "spring-boot.build-image.builder") String imageBuilder; /** * Alias for {@link Image#trustBuilder} to support configuration through command-line * property. */ - @Parameter(property = "spring-boot.build-image.trustBuilder", readonly = true) + @Parameter(property = "spring-boot.build-image.trustBuilder") Boolean trustBuilder; /** @@ -130,7 +130,7 @@ public abstract class BuildImageMojo extends AbstractPackagerMojo { * property. * @since 2.3.1 */ - @Parameter(property = "spring-boot.build-image.runImage", readonly = true) + @Parameter(property = "spring-boot.build-image.runImage") String runImage; /** @@ -138,21 +138,21 @@ public abstract class BuildImageMojo extends AbstractPackagerMojo { * property. * @since 2.4.0 */ - @Parameter(property = "spring-boot.build-image.cleanCache", readonly = true) + @Parameter(property = "spring-boot.build-image.cleanCache") Boolean cleanCache; /** * Alias for {@link Image#pullPolicy} to support configuration through command-line * property. */ - @Parameter(property = "spring-boot.build-image.pullPolicy", readonly = true) + @Parameter(property = "spring-boot.build-image.pullPolicy") PullPolicy pullPolicy; /** * Alias for {@link Image#publish} to support configuration through command-line * property. */ - @Parameter(property = "spring-boot.build-image.publish", readonly = true) + @Parameter(property = "spring-boot.build-image.publish") Boolean publish; /** @@ -160,7 +160,7 @@ public abstract class BuildImageMojo extends AbstractPackagerMojo { * property. * @since 2.6.0 */ - @Parameter(property = "spring-boot.build-image.network", readonly = true) + @Parameter(property = "spring-boot.build-image.network") String network; /** @@ -168,7 +168,7 @@ public abstract class BuildImageMojo extends AbstractPackagerMojo { * property. * @since 3.1.0 */ - @Parameter(property = "spring-boot.build-image.createdDate", readonly = true) + @Parameter(property = "spring-boot.build-image.createdDate") String createdDate; /** @@ -176,7 +176,7 @@ public abstract class BuildImageMojo extends AbstractPackagerMojo { * command-line property. * @since 3.1.0 */ - @Parameter(property = "spring-boot.build-image.applicationDirectory", readonly = true) + @Parameter(property = "spring-boot.build-image.applicationDirectory") String applicationDirectory; /** @@ -184,7 +184,7 @@ public abstract class BuildImageMojo extends AbstractPackagerMojo { * property. * @since 3.4.0 */ - @Parameter(property = "spring-boot.build-image.imagePlatform", readonly = true) + @Parameter(property = "spring-boot.build-image.imagePlatform") String imagePlatform; /** diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/FilterableDependency.java b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/FilterableDependency.java index 2d12921fa055..f95d895909ec 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/FilterableDependency.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/FilterableDependency.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -79,7 +79,7 @@ void setClassifier(String classifier) { public void set(String property) { String[] parts = property.split(":"); Assert.isTrue(parts.length == 2 || parts.length == 3, getClass().getSimpleName() - + " must be in the form groupId:artifactId or groupId:artifactId:classifier"); + + " 'property' must be in the form groupId:artifactId or groupId:artifactId:classifier"); setGroupId(parts[0]); setArtifactId(parts[1]); if (parts.length == 3) { diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/Layers.java b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/Layers.java index cb4d6a235d88..6cdd829e9455 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/Layers.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/java/org/springframework/boot/maven/Layers.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,9 +28,6 @@ public class Layers { private boolean enabled = true; - @Deprecated(since = "3.3.0", forRemoval = true) - private boolean includeLayerTools = true; - private File configuration; /** @@ -41,16 +38,6 @@ public boolean isEnabled() { return this.enabled; } - /** - * Whether to include the layer tools jar. - * @return true if layer tools should be included - * @deprecated since 3.3.0 for removal in 3.5.0 in favor of {@code includeTools}. - */ - @Deprecated(since = "3.3.0", forRemoval = true) - public boolean isIncludeLayerTools() { - return this.includeLayerTools; - } - /** * The location of the layers configuration file. If no file is provided, a default * configuration is used with four layers: {@code application}, {@code resources}, diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/xsd/layers-3.5.xsd b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/xsd/layers-3.5.xsd new file mode 100644 index 000000000000..20219b9bd8b1 --- /dev/null +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/main/xsd/layers-3.5.xsd @@ -0,0 +1,100 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/java/org/springframework/boot/maven/ImageTests.java b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/java/org/springframework/boot/maven/ImageTests.java index 12b376054365..126fec386649 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/java/org/springframework/boot/maven/ImageTests.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/java/org/springframework/boot/maven/ImageTests.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/dependencies-layer-no-filter.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/dependencies-layer-no-filter.xml index e5014dce4393..4a65ffd54c6f 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/dependencies-layer-no-filter.xml +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/dependencies-layer-no-filter.xml @@ -1,7 +1,7 @@ + https://www.springframework.org/schema/boot/layers/layers-3.5.xsd"> diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/layers.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/layers.xml index 7f12e4fc63d8..731fd8ddfd43 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/layers.xml +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/layers.xml @@ -1,7 +1,7 @@ + https://www.springframework.org/schema/boot/layers/layers-3.5.xsd"> META-INF/resources/** diff --git a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/resource-layer-no-filter.xml b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/resource-layer-no-filter.xml index bb698a6323e5..76d65328d8eb 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/resource-layer-no-filter.xml +++ b/spring-boot-project/spring-boot-tools/spring-boot-maven-plugin/src/test/resources/resource-layer-no-filter.xml @@ -1,7 +1,7 @@ + https://www.springframework.org/schema/boot/layers/layers-3.5.xsd"> diff --git a/spring-boot-project/spring-boot-tools/spring-boot-properties-migrator/src/main/java/org/springframework/boot/context/properties/migrator/PropertyMigration.java b/spring-boot-project/spring-boot-tools/spring-boot-properties-migrator/src/main/java/org/springframework/boot/context/properties/migrator/PropertyMigration.java index 933adf52121b..61a5223e57ca 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-properties-migrator/src/main/java/org/springframework/boot/context/properties/migrator/PropertyMigration.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-properties-migrator/src/main/java/org/springframework/boot/context/properties/migrator/PropertyMigration.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -154,7 +154,7 @@ private static ConfigurationPropertyName getNewMapPropertyName(ConfigurationProp ConfigurationPropertyName oldName = property.getName(); ConfigurationPropertyName oldPrefix = ConfigurationPropertyName.of(metadata.getId()); Assert.state(oldPrefix.isAncestorOf(oldName), - String.format("'%s' is not an ancestor of '%s'", oldPrefix, oldName)); + () -> "'%s' is not an ancestor of '%s'".formatted(oldPrefix, oldName)); ConfigurationPropertyName newPrefix = ConfigurationPropertyName.of(replacement.getId()); return newPrefix.append(oldName.subName(oldPrefix.getNumberOfElements())); } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/build.gradle b/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/build.gradle index aae510e78bc2..a14cdea6199d 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/build.gradle +++ b/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/build.gradle @@ -21,6 +21,7 @@ dependencies { optional("org.testcontainers:grafana") optional("org.testcontainers:junit-jupiter") optional("org.testcontainers:kafka") + optional("org.testcontainers:ldap") optional("org.testcontainers:mongodb") optional("org.testcontainers:neo4j") optional("org.testcontainers:oracle-xe") diff --git a/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/src/main/java/org/springframework/boot/testsupport/container/ElasticsearchContainer8.java b/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/src/main/java/org/springframework/boot/testsupport/container/ElasticsearchContainer8.java new file mode 100644 index 000000000000..587849e4fc76 --- /dev/null +++ b/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/src/main/java/org/springframework/boot/testsupport/container/ElasticsearchContainer8.java @@ -0,0 +1,34 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.testsupport.container; + +import org.testcontainers.elasticsearch.ElasticsearchContainer; + +/** + * A container suitable for testing Elasticsearch 8. + * + * @author Dmytro Nosan + */ +public class ElasticsearchContainer8 extends ElasticsearchContainer { + + public ElasticsearchContainer8() { + super(TestImage.ELASTICSEARCH_8.toString()); + addEnv("ES_JAVA_OPTS", "-Xms32m -Xmx512m"); + addEnv("xpack.security.enabled", "false"); + } + +} diff --git a/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/src/main/java/org/springframework/boot/testsupport/container/TestImage.java b/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/src/main/java/org/springframework/boot/testsupport/container/TestImage.java index 924c67342184..cd59344757fb 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/src/main/java/org/springframework/boot/testsupport/container/TestImage.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-test-support-docker/src/main/java/org/springframework/boot/testsupport/container/TestImage.java @@ -39,6 +39,7 @@ import org.testcontainers.elasticsearch.ElasticsearchContainer; import org.testcontainers.grafana.LgtmStackContainer; import org.testcontainers.kafka.ConfluentKafkaContainer; +import org.testcontainers.ldap.LLdapContainer; import org.testcontainers.redpanda.RedpandaContainer; import org.testcontainers.utility.DockerImageName; @@ -86,7 +87,7 @@ public enum TestImage { /** * A container image suitable for testing Cassandra using the deprecated * {@link org.testcontainers.containers.CassandraContainer}. - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of {@link #CASSANDRA} + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of {@link #CASSANDRA} */ @SuppressWarnings("deprecation") @Deprecated(since = "3.4.0", forRemoval = true) @@ -117,7 +118,7 @@ public enum TestImage { /** * A container image suitable for testing Elasticsearch 8. */ - ELASTICSEARCH_8("elasticsearch", "8.6.1"), + ELASTICSEARCH_8("elasticsearch", "8.17.1"), /** * A container image suitable for testing Grafana OTel LGTM. @@ -128,7 +129,7 @@ public enum TestImage { /** * A container image suitable for testing Hazelcast. */ - HAZELCAST("hazelcast/hazelcast", "5.5.0-slim", () -> HazelcastContainer.class), + HAZELCAST("hazelcast/hazelcast", "5.5.0-slim-jdk17", () -> HazelcastContainer.class), /** * A container image suitable for testing Confluent's distribution of Kafka. @@ -138,13 +139,18 @@ public enum TestImage { /** * A container image suitable for testing Confluent's distribution of Kafka using the * deprecated {@link org.testcontainers.containers.KafkaContainer}. - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of {@link #CONFLUENT_KAFKA} + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of {@link #CONFLUENT_KAFKA} */ @SuppressWarnings("deprecation") @Deprecated(since = "3.4.0", forRemoval = true) CONFLUENT_KAFKA_DEPRECATED("confluentinc/cp-kafka", "7.4.0", () -> org.testcontainers.containers.KafkaContainer.class), + /** + * A container image suitable for testing LLDAP. + */ + LLDAP("lldap/lldap", "v0.6.1-alpine", () -> LLdapContainer.class), + /** * A container image suitable for testing OpenLDAP. */ diff --git a/spring-boot-project/spring-boot-tools/spring-boot-test-support/src/main/java/org/springframework/boot/testsupport/classpath/resources/ResourcesExtension.java b/spring-boot-project/spring-boot-tools/spring-boot-test-support/src/main/java/org/springframework/boot/testsupport/classpath/resources/ResourcesExtension.java index 4d0d9ab2c981..fa61100ccc70 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-test-support/src/main/java/org/springframework/boot/testsupport/classpath/resources/ResourcesExtension.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-test-support/src/main/java/org/springframework/boot/testsupport/classpath/resources/ResourcesExtension.java @@ -38,7 +38,6 @@ import org.junit.jupiter.api.extension.ParameterResolutionException; import org.junit.jupiter.api.extension.ParameterResolver; import org.junit.platform.commons.support.AnnotationSupport; -import org.junit.platform.commons.support.SearchOption; import org.springframework.util.StreamUtils; @@ -66,7 +65,7 @@ public void beforeEach(ExtensionContext context) throws Exception { resourcesOf(testMethod) .forEach((resource) -> resources.addResource(resource.name(), resource.content(), resource.additional())); resourceDirectoriesOf(testMethod).forEach((directory) -> resources.addDirectory(directory.value())); - packageResourcesOf(testMethod).forEach((withPackageResources) -> resources + packageResourcesOf(testMethod, context).forEach((withPackageResources) -> resources .addPackage(testMethod.getDeclaringClass().getPackage(), withPackageResources.value())); ResourcesClassLoader classLoader = new ResourcesClassLoader(context.getRequiredTestClass().getClassLoader(), resources); @@ -93,12 +92,11 @@ private List withAnnotationsOf(Method method, Class return annotations; } - private List packageResourcesOf(Method method) { + private List packageResourcesOf(Method method, ExtensionContext context) { List annotations = new ArrayList<>(); AnnotationSupport.findAnnotation(method, WithPackageResources.class).ifPresent(annotations::add); AnnotationSupport - .findAnnotation(method.getDeclaringClass(), WithPackageResources.class, - SearchOption.INCLUDE_ENCLOSING_CLASSES) + .findAnnotation(method.getDeclaringClass(), WithPackageResources.class, context.getEnclosingTestClasses()) .ifPresent(annotations::add); return annotations; } diff --git a/spring-boot-project/spring-boot-tools/spring-boot-test-support/src/main/java/org/springframework/boot/testsupport/process/DisabledIfProcessUnavailableCondition.java b/spring-boot-project/spring-boot-tools/spring-boot-test-support/src/main/java/org/springframework/boot/testsupport/process/DisabledIfProcessUnavailableCondition.java index 612124659975..2b5434139714 100644 --- a/spring-boot-project/spring-boot-tools/spring-boot-test-support/src/main/java/org/springframework/boot/testsupport/process/DisabledIfProcessUnavailableCondition.java +++ b/spring-boot-project/spring-boot-tools/spring-boot-test-support/src/main/java/org/springframework/boot/testsupport/process/DisabledIfProcessUnavailableCondition.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -69,7 +69,7 @@ private void check(String[] command) { ProcessBuilder processBuilder = new ProcessBuilder(command); try { Process process = processBuilder.start(); - Assert.isTrue(process.waitFor(30, TimeUnit.SECONDS), "Process did not exit within 30 seconds"); + Assert.state(process.waitFor(30, TimeUnit.SECONDS), "Process did not exit within 30 seconds"); Assert.state(process.exitValue() == 0, () -> "Process exited with %d".formatted(process.exitValue())); process.destroy(); } diff --git a/spring-boot-project/spring-boot/build.gradle b/spring-boot-project/spring-boot/build.gradle index 2550468bd776..fb8fff7b9fc9 100644 --- a/spring-boot-project/spring-boot/build.gradle +++ b/spring-boot-project/spring-boot/build.gradle @@ -48,6 +48,7 @@ dependencies { exclude(group: "commons-logging", module: "commons-logging") } optional("org.apache.httpcomponents.client5:httpclient5") + optional("org.apache.httpcomponents.core5:httpcore5-reactive") optional("org.apache.logging.log4j:log4j-api") optional("org.apache.logging.log4j:log4j-core") optional("org.apache.logging.log4j:log4j-jul") @@ -60,6 +61,7 @@ dependencies { optional("org.crac:crac") optional("org.eclipse.jetty:jetty-alpn-conscrypt-server") optional("org.eclipse.jetty:jetty-client") + optional("org.eclipse.jetty:jetty-reactive-httpclient") optional("org.eclipse.jetty:jetty-util") optional("org.eclipse.jetty.ee10:jetty-ee10-servlets") optional("org.eclipse.jetty.ee10:jetty-ee10-webapp") @@ -97,9 +99,13 @@ dependencies { exclude group: "org.eclipse.jetty", module: "jetty-servlet" exclude group: "jakarta.mail", module: "jakarta.mail-api" } + optional("org.vibur:vibur-dbcp") optional("org.yaml:snakeyaml") optional("org.jetbrains.kotlin:kotlin-reflect") optional("org.jetbrains.kotlin:kotlin-stdlib") + optional("software.amazon.jdbc:aws-advanced-jdbc-wrapper") { + exclude(group: "commons-logging", module: "commons-logging") + } testFixturesCompileOnly("jakarta.servlet:jakarta.servlet-api") testFixturesCompileOnly("org.mockito:mockito-core") diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ApplicationInfoPropertySource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ApplicationInfoPropertySource.java index d4b7e62e4451..7f3dc102608d 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ApplicationInfoPropertySource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ApplicationInfoPropertySource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,8 @@ import java.util.HashMap; import java.util.Map; +import org.springframework.boot.origin.Origin; +import org.springframework.boot.origin.OriginLookup; import org.springframework.boot.system.ApplicationPid; import org.springframework.core.env.ConfigurableEnvironment; import org.springframework.core.env.MapPropertySource; @@ -32,7 +34,7 @@ * * @author Moritz Halbritter */ -class ApplicationInfoPropertySource extends MapPropertySource { +class ApplicationInfoPropertySource extends MapPropertySource implements OriginLookup { static final String NAME = "applicationInfo"; @@ -44,6 +46,16 @@ class ApplicationInfoPropertySource extends MapPropertySource { super(NAME, getProperties(applicationVersion)); } + @Override + public Origin getOrigin(String key) { + return null; + } + + @Override + public boolean isImmutable() { + return true; + } + private static Map getProperties(String applicationVersion) { Map result = new HashMap<>(); if (StringUtils.hasText(applicationVersion)) { diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/BeanDefinitionLoader.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/BeanDefinitionLoader.java index db52cea7ff0e..4e0004fff39f 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/BeanDefinitionLoader.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/BeanDefinitionLoader.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -81,8 +81,8 @@ class BeanDefinitionLoader { * @param sources the bean sources */ BeanDefinitionLoader(BeanDefinitionRegistry registry, Object... sources) { - Assert.notNull(registry, "Registry must not be null"); - Assert.notEmpty(sources, "Sources must not be empty"); + Assert.notNull(registry, "'registry' must not be null"); + Assert.notEmpty(sources, "'sources' must not be empty"); this.sources = sources; this.annotatedReader = new AnnotatedBeanDefinitionReader(registry); this.xmlReader = new XmlBeanDefinitionReader(registry); @@ -131,7 +131,7 @@ void load() { } private void load(Object source) { - Assert.notNull(source, "Source must not be null"); + Assert.notNull(source, "'source' must not be null"); if (source instanceof Class type) { load(type); return; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/BootstrapRegistry.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/BootstrapRegistry.java index b983d03f4188..f7d42727073a 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/BootstrapRegistry.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/BootstrapRegistry.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -121,7 +121,7 @@ default Scope getScope() { * @since 2.4.2 */ default InstanceSupplier withScope(Scope scope) { - Assert.notNull(scope, "Scope must not be null"); + Assert.notNull(scope, "'scope' must not be null"); InstanceSupplier parent = this; return new InstanceSupplier<>() { diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/DefaultApplicationArguments.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/DefaultApplicationArguments.java index 8298411e5631..f84eaa92d3fe 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/DefaultApplicationArguments.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/DefaultApplicationArguments.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -38,7 +38,7 @@ public class DefaultApplicationArguments implements ApplicationArguments { private final String[] args; public DefaultApplicationArguments(String... args) { - Assert.notNull(args, "Args must not be null"); + Assert.notNull(args, "'args' must not be null"); this.source = new Source(args); this.args = args; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/DefaultBootstrapContext.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/DefaultBootstrapContext.java index eec1b2a714cd..73301dec307f 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/DefaultBootstrapContext.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/DefaultBootstrapContext.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -52,8 +52,8 @@ public void registerIfAbsent(Class type, InstanceSupplier instanceSupp } private void register(Class type, InstanceSupplier instanceSupplier, boolean replaceExisting) { - Assert.notNull(type, "Type must not be null"); - Assert.notNull(instanceSupplier, "InstanceSupplier must not be null"); + Assert.notNull(type, "'type' must not be null"); + Assert.notNull(instanceSupplier, "'instanceSupplier' must not be null"); synchronized (this.instanceSuppliers) { boolean alreadyRegistered = this.instanceSuppliers.containsKey(type); if (replaceExisting || !alreadyRegistered) { diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ExitCodeGenerators.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ExitCodeGenerators.java index 613143a09b62..edbccab0978a 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ExitCodeGenerators.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ExitCodeGenerators.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,39 +42,39 @@ class ExitCodeGenerators implements Iterable { private final List generators = new ArrayList<>(); void addAll(Throwable exception, ExitCodeExceptionMapper... mappers) { - Assert.notNull(exception, "Exception must not be null"); - Assert.notNull(mappers, "Mappers must not be null"); + Assert.notNull(exception, "'exception' must not be null"); + Assert.notNull(mappers, "'mappers' must not be null"); addAll(exception, Arrays.asList(mappers)); } void addAll(Throwable exception, Iterable mappers) { - Assert.notNull(exception, "Exception must not be null"); - Assert.notNull(mappers, "Mappers must not be null"); + Assert.notNull(exception, "'exception' must not be null"); + Assert.notNull(mappers, "'mappers' must not be null"); for (ExitCodeExceptionMapper mapper : mappers) { add(exception, mapper); } } void add(Throwable exception, ExitCodeExceptionMapper mapper) { - Assert.notNull(exception, "Exception must not be null"); - Assert.notNull(mapper, "Mapper must not be null"); + Assert.notNull(exception, "'exception' must not be null"); + Assert.notNull(mapper, "'mapper' must not be null"); add(new MappedExitCodeGenerator(exception, mapper)); } void addAll(ExitCodeGenerator... generators) { - Assert.notNull(generators, "Generators must not be null"); + Assert.notNull(generators, "'generators' must not be null"); addAll(Arrays.asList(generators)); } void addAll(Iterable generators) { - Assert.notNull(generators, "Generators must not be null"); + Assert.notNull(generators, "'generators' must not be null"); for (ExitCodeGenerator generator : generators) { add(generator); } } void add(ExitCodeGenerator generator) { - Assert.notNull(generator, "Generator must not be null"); + Assert.notNull(generator, "'generator' must not be null"); this.generators.add(generator); AnnotationAwareOrderComparator.sort(this.generators); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ResourceBanner.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ResourceBanner.java index 5db3c468d2fc..c4902fc8c017 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ResourceBanner.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ResourceBanner.java @@ -58,8 +58,8 @@ public class ResourceBanner implements Banner { private final Resource resource; public ResourceBanner(Resource resource) { - Assert.notNull(resource, "Resource must not be null"); - Assert.isTrue(resource.exists(), "Resource must exist"); + Assert.notNull(resource, "'resource' must not be null"); + Assert.isTrue(resource.exists(), "'resource' must exist"); this.resource = resource; } @@ -155,7 +155,7 @@ private Map getVersionsMap(Class sourceClass, Environment env * Returns the application version. * @param sourceClass the source class * @return the application version or {@code null} if unknown - * @deprecated since 3.4.0 for removal in 3.6.0 + * @deprecated since 3.4.0 for removal in 4.0.0 */ @Deprecated(since = "3.4.0", forRemoval = true) protected String getApplicationVersion(Class sourceClass) { diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplication.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplication.java index e1d311ff765e..f87954d8ef1b 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplication.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplication.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -270,7 +270,7 @@ public SpringApplication(Class... primarySources) { @SuppressWarnings({ "unchecked", "rawtypes" }) public SpringApplication(ResourceLoader resourceLoader, Class... primarySources) { this.resourceLoader = resourceLoader; - Assert.notNull(primarySources, "PrimarySources must not be null"); + Assert.notNull(primarySources, "'primarySources' must not be null"); this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources)); this.properties.setWebApplicationType(WebApplicationType.deduceFromClasspath()); this.bootstrapRegistryInitializers = new ArrayList<>( @@ -410,7 +410,7 @@ private void prepareContext(DefaultBootstrapContext bootstrapContext, Configurab if (!AotDetector.useGeneratedArtifacts()) { // Load the sources Set sources = getAllSources(); - Assert.notEmpty(sources, "Sources must not be empty"); + Assert.state(!ObjectUtils.isEmpty(sources), "No sources defined"); load(context, sources.toArray(new Object[0])); } listeners.contextLoaded(context); @@ -608,7 +608,7 @@ protected void applyInitializers(ConfigurableApplicationContext context) { for (ApplicationContextInitializer initializer : getInitializers()) { Class requiredType = GenericTypeResolver.resolveTypeArgument(initializer.getClass(), ApplicationContextInitializer.class); - Assert.isInstanceOf(requiredType, context, "Unable to call initializer."); + Assert.state(requiredType.isInstance(context), "Unable to call initializer"); initializer.initialize(context); } } @@ -630,7 +630,7 @@ protected void logStartupInfo(ConfigurableApplicationContext context) { * Called to log startup information, subclasses may override to add additional * logging. * @param isRoot true if this application is the root of a context hierarchy - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link #logStartupInfo(ConfigurableApplicationContext)} */ @Deprecated(since = "3.4.0", forRemoval = true) @@ -959,7 +959,7 @@ public WebApplicationType getWebApplicationType() { * @since 2.0.0 */ public void setWebApplicationType(WebApplicationType webApplicationType) { - Assert.notNull(webApplicationType, "WebApplicationType must not be null"); + Assert.notNull(webApplicationType, "'webApplicationType' must not be null"); this.properties.setWebApplicationType(webApplicationType); } @@ -1068,7 +1068,7 @@ public void setAddConversionService(boolean addConversionService) { * @since 2.4.5 */ public void addBootstrapRegistryInitializer(BootstrapRegistryInitializer bootstrapRegistryInitializer) { - Assert.notNull(bootstrapRegistryInitializer, "BootstrapRegistryInitializer must not be null"); + Assert.notNull(bootstrapRegistryInitializer, "'bootstrapRegistryInitializer' must not be null"); this.bootstrapRegistryInitializers.addAll(Arrays.asList(bootstrapRegistryInitializer)); } @@ -1169,7 +1169,7 @@ public Set getSources() { * @see #getAllSources() */ public void setSources(Set sources) { - Assert.notNull(sources, "Sources must not be null"); + Assert.notNull(sources, "'sources' must not be null"); this.properties.setSources(sources); } @@ -1196,7 +1196,7 @@ public Set getAllSources() { * @param resourceLoader the resource loader */ public void setResourceLoader(ResourceLoader resourceLoader) { - Assert.notNull(resourceLoader, "ResourceLoader must not be null"); + Assert.notNull(resourceLoader, "'resourceLoader' must not be null"); this.resourceLoader = resourceLoader; } @@ -1390,7 +1390,7 @@ public static void main(String[] args) throws Exception { * @return the outcome (0 if successful) */ public static int exit(ApplicationContext context, ExitCodeGenerator... exitCodeGenerators) { - Assert.notNull(context, "Context must not be null"); + Assert.notNull(context, "'context' must not be null"); int exitCode = 0; try { try { @@ -1426,7 +1426,7 @@ public static int exit(ApplicationContext context, ExitCodeGenerator... exitCode * @see #withHook(SpringApplicationHook, Runnable) */ public static SpringApplication.Augmented from(ThrowingConsumer main) { - Assert.notNull(main, "Main must not be null"); + Assert.notNull(main, "'main' must not be null"); return new Augmented(main, Collections.emptySet(), Collections.emptySet()); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplicationAotProcessor.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplicationAotProcessor.java index 6eec7ad6c2df..448eb9c3ef12 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplicationAotProcessor.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplicationAotProcessor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -65,8 +65,8 @@ protected GenericApplicationContext prepareApplicationContext(Class applicati public static void main(String[] args) throws Exception { int requiredArgs = 6; - Assert.isTrue(args.length >= requiredArgs, () -> "Usage: " + SpringApplicationAotProcessor.class.getName() - + " "); + Assert.state(args.length >= requiredArgs, () -> "Usage: " + SpringApplicationAotProcessor.class.getName() + + " "); Class application = Class.forName(args[0]); Settings settings = Settings.builder() .sourceOutput(Paths.get(args[1])) @@ -110,7 +110,7 @@ private GenericApplicationContext run(ThrowingSupplier action) { } catch (AbandonedRunException ex) { ApplicationContext context = ex.getApplicationContext(); - Assert.isInstanceOf(GenericApplicationContext.class, context, + Assert.state(context instanceof GenericApplicationContext, () -> "AOT processing requires a GenericApplicationContext but got a " + context.getClass().getName()); return (GenericApplicationContext) context; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplicationShutdownHook.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplicationShutdownHook.java index 433087a1bed9..1347bb80f9fa 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplicationShutdownHook.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/SpringApplicationShutdownHook.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -177,7 +177,7 @@ private final class Handlers implements SpringApplicationShutdownHandlers, Runna @Override public void add(Runnable action) { - Assert.notNull(action, "Action must not be null"); + Assert.notNull(action, "'action' must not be null"); addRuntimeShutdownHookIfNecessary(); synchronized (SpringApplicationShutdownHook.class) { assertNotInProgress(); @@ -187,7 +187,7 @@ public void add(Runnable action) { @Override public void remove(Runnable action) { - Assert.notNull(action, "Action must not be null"); + Assert.notNull(action, "'action' must not be null"); synchronized (SpringApplicationShutdownHook.class) { assertNotInProgress(); this.actions.remove(new Handler(action)); diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/StartupInfoLogger.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/StartupInfoLogger.java index b3d40bd8085b..35bb085afe3c 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/StartupInfoLogger.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/StartupInfoLogger.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,7 +49,7 @@ class StartupInfoLogger { } void logStarting(Log applicationLog) { - Assert.notNull(applicationLog, "Log must not be null"); + Assert.notNull(applicationLog, "'applicationLog' must not be null"); applicationLog.info(LogMessage.of(this::getStartingMessage)); applicationLog.debug(LogMessage.of(this::getRunningMessage)); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/admin/SpringApplicationAdminMXBeanRegistrar.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/admin/SpringApplicationAdminMXBeanRegistrar.java index 71ebfd98353f..ac7f33a84f99 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/admin/SpringApplicationAdminMXBeanRegistrar.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/admin/SpringApplicationAdminMXBeanRegistrar.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -72,7 +72,7 @@ public SpringApplicationAdminMXBeanRegistrar(String name) throws MalformedObject @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { Assert.state(applicationContext instanceof ConfigurableApplicationContext, - "ApplicationContext does not implement ConfigurableApplicationContext"); + "'applicationContext' must be a ConfigurableApplicationContext"); this.applicationContext = (ConfigurableApplicationContext) applicationContext; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ansi/Ansi8BitColor.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ansi/Ansi8BitColor.java index 2b926383f803..50e30b7cab09 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ansi/Ansi8BitColor.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ansi/Ansi8BitColor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -40,7 +40,7 @@ public final class Ansi8BitColor implements AnsiElement { * @throws IllegalArgumentException if color code is not between 0 and 255. */ private Ansi8BitColor(String prefix, int code) { - Assert.isTrue(code >= 0 && code <= 255, "Code must be between 0 and 255"); + Assert.isTrue(code >= 0 && code <= 255, "'code' must be between 0 and 255"); this.prefix = prefix; this.code = code; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ansi/AnsiOutput.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ansi/AnsiOutput.java index b0c815168f4a..96a303335295 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ansi/AnsiOutput.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/ansi/AnsiOutput.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -54,7 +54,7 @@ public abstract class AnsiOutput { * @param enabled if ANSI is enabled, disabled or detected */ public static void setEnabled(Enabled enabled) { - Assert.notNull(enabled, "Enabled must not be null"); + Assert.notNull(enabled, "'enabled' must not be null"); AnsiOutput.enabled = enabled; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/availability/ApplicationAvailabilityBean.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/availability/ApplicationAvailabilityBean.java index e88e18ea0054..53f1f2c93a6e 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/availability/ApplicationAvailabilityBean.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/availability/ApplicationAvailabilityBean.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -52,8 +52,8 @@ public ApplicationAvailabilityBean() { @Override public S getState(Class stateType, S defaultState) { - Assert.notNull(stateType, "StateType must not be null"); - Assert.notNull(defaultState, "DefaultState must not be null"); + Assert.notNull(stateType, "'stateType' must not be null"); + Assert.notNull(defaultState, "'defaultState' must not be null"); S state = getState(stateType); return (state != null) ? state : defaultState; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/availability/AvailabilityChangeEvent.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/availability/AvailabilityChangeEvent.java index 19250603a297..49a416784634 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/availability/AvailabilityChangeEvent.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/availability/AvailabilityChangeEvent.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -74,7 +74,7 @@ private Class getStateType() { * @param state the changed availability state */ public static void publish(ApplicationContext context, S state) { - Assert.notNull(context, "Context must not be null"); + Assert.notNull(context, "'context' must not be null"); publish(context, context, state); } @@ -88,7 +88,7 @@ public static void publish(ApplicationContext cont */ public static void publish(ApplicationEventPublisher publisher, Object source, S state) { - Assert.notNull(publisher, "Publisher must not be null"); + Assert.notNull(publisher, "'publisher' must not be null"); publisher.publishEvent(new AvailabilityChangeEvent<>(source, state)); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/ApplicationPidFileWriter.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/ApplicationPidFileWriter.java index 235027a9ecc9..5e5267b4219f 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/ApplicationPidFileWriter.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/ApplicationPidFileWriter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -115,7 +115,7 @@ public ApplicationPidFileWriter(String filename) { * @param file the file containing pid */ public ApplicationPidFileWriter(File file) { - Assert.notNull(file, "File must not be null"); + Assert.notNull(file, "'file' must not be null"); this.file = file; } @@ -128,7 +128,7 @@ public ApplicationPidFileWriter(File file) { * @param triggerEventType the trigger event type */ public void setTriggerEventType(Class triggerEventType) { - Assert.notNull(triggerEventType, "Trigger event type must not be null"); + Assert.notNull(triggerEventType, "'triggerEventType' must not be null"); this.triggerEventType = triggerEventType; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/annotation/Configurations.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/annotation/Configurations.java index 744c04d3f4a4..707b2367a0cb 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/annotation/Configurations.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/annotation/Configurations.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -73,7 +73,7 @@ public abstract class Configurations { * @param classes the configuration classes */ protected Configurations(Collection> classes) { - Assert.notNull(classes, "Classes must not be null"); + Assert.notNull(classes, "'classes' must not be null"); Collection> sorted = sort(classes); this.sorter = null; this.classes = Collections.unmodifiableSet(new LinkedHashSet<>(sorted)); @@ -89,7 +89,7 @@ protected Configurations(Collection> classes) { */ protected Configurations(UnaryOperator>> sorter, Collection> classes, Function, String> beanNameGenerator) { - Assert.notNull(classes, "Classes must not be null"); + Assert.notNull(classes, "'classes' must not be null"); this.sorter = (sorter != null) ? sorter : UnaryOperator.identity(); Collection> sorted = this.sorter.apply(classes); this.classes = Collections.unmodifiableSet(new LinkedHashSet<>(sorted)); @@ -104,7 +104,7 @@ protected final Set> getClasses() { * Sort configuration classes into the order that they should be applied. * @param classes the classes to sort * @return a sorted set of classes - * @deprecated since 3.4.0 for removal in 3.6.0 in favor of + * @deprecated since 3.4.0 for removal in 4.0.0 in favor of * {@link #Configurations(UnaryOperator, Collection, Function)} */ @Deprecated(since = "3.4.0", forRemoval = true) diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigData.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigData.java index 9482c48afe13..31dde9c83ea2 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigData.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigData.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -72,8 +72,8 @@ public ConfigData(Collection> propertySources, Optio */ public ConfigData(Collection> propertySources, PropertySourceOptions propertySourceOptions) { - Assert.notNull(propertySources, "PropertySources must not be null"); - Assert.notNull(propertySourceOptions, "PropertySourceOptions must not be null"); + Assert.notNull(propertySources, "'propertySources' must not be null"); + Assert.notNull(propertySourceOptions, "'propertySourceOptions' must not be null"); this.propertySources = Collections.unmodifiableList(new ArrayList<>(propertySources)); this.propertySourceOptions = propertySourceOptions; } @@ -250,7 +250,7 @@ private Options copy(Consumer> processor) { * @return a new {@link Options} instance */ public static Options of(Option... options) { - Assert.notNull(options, "Options must not be null"); + Assert.notNull(options, "'options' must not be null"); if (options.length == 0) { return NONE; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigDataLocationNotFoundException.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigDataLocationNotFoundException.java index 62bd68dc528b..33f21c68caf9 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigDataLocationNotFoundException.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigDataLocationNotFoundException.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,7 +56,7 @@ public ConfigDataLocationNotFoundException(ConfigDataLocation location, Throwabl */ public ConfigDataLocationNotFoundException(ConfigDataLocation location, String message, Throwable cause) { super(message, cause); - Assert.notNull(location, "Location must not be null"); + Assert.notNull(location, "'location' must not be null"); this.location = location; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigDataResourceNotFoundException.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigDataResourceNotFoundException.java index 0b7d8990a4bd..86a68546b2ea 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigDataResourceNotFoundException.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigDataResourceNotFoundException.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -57,7 +57,7 @@ public ConfigDataResourceNotFoundException(ConfigDataResource resource, Throwabl private ConfigDataResourceNotFoundException(ConfigDataResource resource, ConfigDataLocation location, Throwable cause) { super(getMessage(resource, location), cause); - Assert.notNull(resource, "Resource must not be null"); + Assert.notNull(resource, "'resource' must not be null"); this.resource = resource; this.location = location; } @@ -116,7 +116,7 @@ private static String getReferenceDescription(ConfigDataResource resource, Confi * @param pathToCheck the path to check */ public static void throwIfDoesNotExist(ConfigDataResource resource, Path pathToCheck) { - throwIfDoesNotExist(resource, Files.exists(pathToCheck)); + throwIfNot(resource, Files.exists(pathToCheck)); } /** @@ -126,7 +126,7 @@ public static void throwIfDoesNotExist(ConfigDataResource resource, Path pathToC * @param fileToCheck the file to check */ public static void throwIfDoesNotExist(ConfigDataResource resource, File fileToCheck) { - throwIfDoesNotExist(resource, fileToCheck.exists()); + throwIfNot(resource, fileToCheck.exists()); } /** @@ -136,11 +136,11 @@ public static void throwIfDoesNotExist(ConfigDataResource resource, File fileToC * @param resourceToCheck the resource to check */ public static void throwIfDoesNotExist(ConfigDataResource resource, Resource resourceToCheck) { - throwIfDoesNotExist(resource, resourceToCheck.exists()); + throwIfNot(resource, resourceToCheck.exists()); } - private static void throwIfDoesNotExist(ConfigDataResource resource, boolean exists) { - if (!exists) { + private static void throwIfNot(ConfigDataResource resource, boolean check) { + if (!check) { throw new ConfigDataResourceNotFoundException(resource); } } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigTreeConfigDataLocationResolver.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigTreeConfigDataLocationResolver.java index 156e4a6c03df..5ed87c79520e 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigTreeConfigDataLocationResolver.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigTreeConfigDataLocationResolver.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -60,7 +60,7 @@ public List resolve(ConfigDataLocationResolverCont } private List resolve(String location) throws IOException { - Assert.isTrue(location.endsWith("/"), + Assert.state(location.endsWith("/"), () -> String.format("Config tree location '%s' must end with '/'", location)); if (!this.resourceLoader.isPattern(location)) { return Collections.singletonList(new ConfigTreeConfigDataResource(location)); diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigTreeConfigDataResource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigTreeConfigDataResource.java index b0977f9c0858..f842d1cedfde 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigTreeConfigDataResource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/ConfigTreeConfigDataResource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,12 +36,12 @@ public class ConfigTreeConfigDataResource extends ConfigDataResource { private final Path path; ConfigTreeConfigDataResource(String path) { - Assert.notNull(path, "Path must not be null"); + Assert.notNull(path, "'path' must not be null"); this.path = Paths.get(path).toAbsolutePath(); } ConfigTreeConfigDataResource(Path path) { - Assert.notNull(path, "Path must not be null"); + Assert.notNull(path, "'path' must not be null"); this.path = path.toAbsolutePath(); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/FileExtensionHint.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/FileExtensionHint.java new file mode 100644 index 000000000000..f57d0b51e231 --- /dev/null +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/FileExtensionHint.java @@ -0,0 +1,82 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.context.config; + +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +/** + * User-provided hint for an otherwise missing file extension. + * + * @author Phillip Webb + */ +final class FileExtensionHint { + + private static final Pattern PATTERN = Pattern.compile("^(.*)\\[(\\.\\w+)](?!\\[)$"); + + private static final FileExtensionHint NONE = new FileExtensionHint(null); + + private final Matcher matcher; + + private FileExtensionHint(Matcher matcher) { + this.matcher = matcher; + } + + /** + * Return {@code true} if the hint is present. + * @return if the hint is present + */ + boolean isPresent() { + return this.matcher != null; + } + + /** + * Return the extension from the hint or return the parameter if the hint is not + * {@link #isPresent() present}. + * @param extension the fallback extension + * @return the extension either from the hint or fallback + */ + String orElse(String extension) { + return (this.matcher != null) ? toString() : extension; + } + + @Override + public String toString() { + return (this.matcher != null) ? this.matcher.group(2) : ""; + } + + /** + * Return the {@link FileExtensionHint} from the given value. + * @param value the source value + * @return the {@link FileExtensionHint} (never {@code null}) + */ + static FileExtensionHint from(String value) { + Matcher matcher = PATTERN.matcher(value); + return (matcher.matches()) ? new FileExtensionHint(matcher) : NONE; + } + + /** + * Remove any hint from the given value. + * @param value the source value + * @return the value without any hint + */ + static String removeFrom(String value) { + Matcher matcher = PATTERN.matcher(value); + return (matcher.matches()) ? matcher.group(1) : value; + } + +} diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/StandardConfigDataLocationResolver.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/StandardConfigDataLocationResolver.java index 5da6bd1bb7a3..31cef797a4b0 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/StandardConfigDataLocationResolver.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/StandardConfigDataLocationResolver.java @@ -26,7 +26,6 @@ import java.util.LinkedHashSet; import java.util.List; import java.util.Set; -import java.util.regex.Matcher; import java.util.regex.Pattern; import java.util.stream.Collectors; @@ -54,6 +53,7 @@ * @author Madhura Bhave * @author Phillip Webb * @author Scott Frederick + * @author Sijun Yang * @since 2.4.0 */ public class StandardConfigDataLocationResolver @@ -67,8 +67,6 @@ public class StandardConfigDataLocationResolver private static final Pattern URL_PREFIX = Pattern.compile("^([a-zA-Z][a-zA-Z0-9*]*?:)(.*$)"); - private static final Pattern EXTENSION_HINT_PATTERN = Pattern.compile("^(.*)\\[(\\.\\w+)](?!\\[)$"); - private static final String NO_PROFILE = null; private final Log logger; @@ -148,6 +146,7 @@ private Set getReferences(ConfigDataLocationResolve @Override public List resolveProfileSpecific(ConfigDataLocationResolverContext context, ConfigDataLocation location, Profiles profiles) { + validateProfiles(profiles); return resolve(getProfileSpecificReferences(context, location.split(), profiles)); } @@ -163,6 +162,27 @@ private Set getProfileSpecificReferences(ConfigData return references; } + private void validateProfiles(Profiles profiles) { + for (String profile : profiles) { + validateProfile(profile); + } + } + + private void validateProfile(String profile) { + Assert.hasText(profile, "'profile' must contain text"); + Assert.state(!profile.startsWith("-") && !profile.startsWith("_"), + () -> String.format("Invalid profile '%s': must not start with '-' or '_'", profile)); + Assert.state(!profile.endsWith("-") && !profile.endsWith("_"), + () -> String.format("Invalid profile '%s': must not end with '-' or '_'", profile)); + profile.codePoints().forEach((codePoint) -> { + if (codePoint == '-' || codePoint == '_' || Character.isLetterOrDigit(codePoint)) { + return; + } + throw new IllegalStateException( + String.format("Invalid profile '%s': must contain only letters, digits, '-', or '_'", profile)); + }); + } + private String getResourceLocation(ConfigDataLocationResolverContext context, ConfigDataLocation configDataLocation) { String resourceLocation = configDataLocation.getNonPrefixedValue(PREFIX); @@ -215,17 +235,16 @@ private Deque getReferencesForConfigName(String nam private Set getReferencesForFile(ConfigDataLocation configDataLocation, String file, String profile) { - Matcher extensionHintMatcher = EXTENSION_HINT_PATTERN.matcher(file); - boolean extensionHintLocation = extensionHintMatcher.matches(); - if (extensionHintLocation) { - file = extensionHintMatcher.group(1) + extensionHintMatcher.group(2); + FileExtensionHint fileExtensionHint = FileExtensionHint.from(file); + if (fileExtensionHint.isPresent()) { + file = FileExtensionHint.removeFrom(file) + fileExtensionHint; } for (PropertySourceLoader propertySourceLoader : this.propertySourceLoaders) { - String extension = getLoadableFileExtension(propertySourceLoader, file); - if (extension != null) { - String root = file.substring(0, file.length() - extension.length() - 1); + String fileExtension = getLoadableFileExtension(propertySourceLoader, file); + if (fileExtension != null) { + String root = file.substring(0, file.length() - fileExtension.length() - 1); StandardConfigDataReference reference = new StandardConfigDataReference(configDataLocation, null, root, - profile, (!extensionHintLocation) ? extension : null, propertySourceLoader); + profile, (!fileExtensionHint.isPresent()) ? fileExtension : null, propertySourceLoader); return Collections.singleton(reference); } } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/StandardConfigDataResource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/StandardConfigDataResource.java index aac0897413eb..f85e56eccc79 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/StandardConfigDataResource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/StandardConfigDataResource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,8 +56,8 @@ public class StandardConfigDataResource extends ConfigDataResource { * @param emptyDirectory if the resource is an empty directory that we know exists */ StandardConfigDataResource(StandardConfigDataReference reference, Resource resource, boolean emptyDirectory) { - Assert.notNull(reference, "Reference must not be null"); - Assert.notNull(resource, "Resource must not be null"); + Assert.notNull(reference, "'reference' must not be null"); + Assert.notNull(resource, "'resource' must not be null"); this.reference = reference; this.resource = resource; this.emptyDirectory = emptyDirectory; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/SystemEnvironmentConfigDataLoader.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/SystemEnvironmentConfigDataLoader.java new file mode 100644 index 000000000000..7b9c78666841 --- /dev/null +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/SystemEnvironmentConfigDataLoader.java @@ -0,0 +1,41 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.context.config; + +import java.io.IOException; +import java.util.List; + +import org.springframework.core.env.PropertySource; + +/** + * {@link ConfigDataLoader} to load data from system environment variables. + * + * @author Moritz Halbritter + */ +class SystemEnvironmentConfigDataLoader implements ConfigDataLoader { + + @Override + public ConfigData load(ConfigDataLoaderContext context, SystemEnvironmentConfigDataResource resource) + throws IOException, ConfigDataResourceNotFoundException { + List> loaded = resource.load(); + if (loaded == null) { + throw new ConfigDataResourceNotFoundException(resource); + } + return new ConfigData(loaded); + } + +} diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/SystemEnvironmentConfigDataLocationResolver.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/SystemEnvironmentConfigDataLocationResolver.java new file mode 100644 index 000000000000..4c3692de6650 --- /dev/null +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/SystemEnvironmentConfigDataLocationResolver.java @@ -0,0 +1,94 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.context.config; + +import java.util.Collections; +import java.util.List; +import java.util.function.Function; + +import org.springframework.boot.env.PropertySourceLoader; +import org.springframework.core.io.support.SpringFactoriesLoader; + +/** + * {@link ConfigDataLocationResolver} to resolve {@code env:} locations. + * + * @author Moritz Halbritter + * @author Phillip Webb + */ +class SystemEnvironmentConfigDataLocationResolver + implements ConfigDataLocationResolver { + + private static final String PREFIX = "env:"; + + private static final String DEFAULT_EXTENSION = ".properties"; + + private final List loaders; + + private final Function environment; + + SystemEnvironmentConfigDataLocationResolver() { + this.loaders = SpringFactoriesLoader.loadFactories(PropertySourceLoader.class, getClass().getClassLoader()); + this.environment = System::getenv; + } + + SystemEnvironmentConfigDataLocationResolver(List loaders, + Function environment) { + this.loaders = loaders; + this.environment = environment; + } + + @Override + public boolean isResolvable(ConfigDataLocationResolverContext context, ConfigDataLocation location) { + return location.hasPrefix(PREFIX); + } + + @Override + public List resolve(ConfigDataLocationResolverContext context, + ConfigDataLocation location) + throws ConfigDataLocationNotFoundException, ConfigDataResourceNotFoundException { + String value = location.getNonPrefixedValue(PREFIX); + FileExtensionHint fileExtensionHint = FileExtensionHint.from(value); + String variableName = FileExtensionHint.removeFrom(value); + PropertySourceLoader loader = getLoader(fileExtensionHint.orElse(DEFAULT_EXTENSION)); + if (hasEnvVariable(variableName)) { + return List.of(new SystemEnvironmentConfigDataResource(variableName, loader, this.environment)); + } + if (location.isOptional()) { + return Collections.emptyList(); + } + throw new ConfigDataLocationNotFoundException(location, + "Environment variable '%s' is not set".formatted(variableName), null); + } + + private PropertySourceLoader getLoader(String extension) { + extension = (!extension.startsWith(".")) ? extension : extension.substring(1); + for (PropertySourceLoader loader : this.loaders) { + for (String supportedExtension : loader.getFileExtensions()) { + if (supportedExtension.equalsIgnoreCase(extension)) { + return loader; + } + } + } + throw new IllegalStateException( + "File extension '%s' is not known to any PropertySourceLoader".formatted(extension)); + } + + private boolean hasEnvVariable(String variableName) { + return this.environment.apply(variableName) != null; + } + +} diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/SystemEnvironmentConfigDataResource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/SystemEnvironmentConfigDataResource.java new file mode 100644 index 000000000000..8c0f7fd12ed9 --- /dev/null +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/config/SystemEnvironmentConfigDataResource.java @@ -0,0 +1,92 @@ +/* + * Copyright 2012-2025 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.boot.context.config; + +import java.io.IOException; +import java.nio.charset.StandardCharsets; +import java.util.List; +import java.util.Objects; +import java.util.function.Function; + +import org.springframework.boot.env.PropertySourceLoader; +import org.springframework.core.env.PropertySource; +import org.springframework.core.io.ByteArrayResource; +import org.springframework.util.ClassUtils; +import org.springframework.util.StringUtils; + +/** + * {@link ConfigDataResource} used by {@link SystemEnvironmentConfigDataLoader}. + * + * @author Moritz Halbritter + */ +class SystemEnvironmentConfigDataResource extends ConfigDataResource { + + private final String variableName; + + private final PropertySourceLoader loader; + + private final Function environment; + + SystemEnvironmentConfigDataResource(String variableName, PropertySourceLoader loader, + Function environment) { + this.variableName = variableName; + this.loader = loader; + this.environment = environment; + } + + String getVariableName() { + return this.variableName; + } + + PropertySourceLoader getLoader() { + return this.loader; + } + + List> load() throws IOException { + String content = this.environment.apply(this.variableName); + return (content != null) ? this.loader.load(StringUtils.capitalize(toString()), asResource(content)) : null; + } + + private ByteArrayResource asResource(String content) { + return new ByteArrayResource(content.getBytes(StandardCharsets.UTF_8)); + } + + @Override + public boolean equals(Object obj) { + if (this == obj) { + return true; + } + if (obj == null || getClass() != obj.getClass()) { + return false; + } + SystemEnvironmentConfigDataResource other = (SystemEnvironmentConfigDataResource) obj; + return Objects.equals(this.loader.getClass(), other.loader.getClass()) + && Objects.equals(this.variableName, other.variableName); + } + + @Override + public int hashCode() { + return Objects.hash(this.variableName, this.loader.getClass()); + } + + @Override + public String toString() { + return "system envionement variable [" + this.variableName + "] content loaded using " + + ClassUtils.getShortName(this.loader.getClass()); + } + +} diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/metrics/buffering/BufferingApplicationStartup.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/metrics/buffering/BufferingApplicationStartup.java index 33b4ab0fcb1a..99639fe781ff 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/metrics/buffering/BufferingApplicationStartup.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/metrics/buffering/BufferingApplicationStartup.java @@ -1,5 +1,5 @@ /* - * Copyright 2002-2021 the original author or authors. + * Copyright 2002-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -91,7 +91,7 @@ public BufferingApplicationStartup(int capacity) { * already. */ public void startRecording() { - Assert.state(this.events.isEmpty(), "Cannot restart recording once steps have been buffered."); + Assert.state(this.events.isEmpty(), "Cannot restart recording once steps have been buffered"); this.startTime = this.clock.instant(); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/BoundConfigurationProperties.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/BoundConfigurationProperties.java index 416c77664834..16d29352c224 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/BoundConfigurationProperties.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/BoundConfigurationProperties.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -77,7 +77,7 @@ public static BoundConfigurationProperties get(ApplicationContext context) { } static void register(BeanDefinitionRegistry registry) { - Assert.notNull(registry, "Registry must not be null"); + Assert.notNull(registry, "'registry' must not be null"); if (!registry.containsBeanDefinition(BEAN_NAME)) { BeanDefinition definition = BeanDefinitionBuilder.rootBeanDefinition(BoundConfigurationProperties.class) .setRole(BeanDefinition.ROLE_INFRASTRUCTURE) diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConfigurationPropertiesBean.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConfigurationPropertiesBean.java index 35a7a71552b7..e0f1118e2df7 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConfigurationPropertiesBean.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConfigurationPropertiesBean.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -128,7 +128,7 @@ public Bindable asBindTarget() { * @return a map of all configuration properties beans keyed by the bean name */ public static Map getAll(ApplicationContext applicationContext) { - Assert.notNull(applicationContext, "ApplicationContext must not be null"); + Assert.notNull(applicationContext, "'applicationContext' must not be null"); if (applicationContext instanceof ConfigurableApplicationContext configurableContext) { return getAll(configurableContext); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConfigurationPropertiesBindingPostProcessor.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConfigurationPropertiesBindingPostProcessor.java index c80c0526e528..34177a1c581e 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConfigurationPropertiesBindingPostProcessor.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConfigurationPropertiesBindingPostProcessor.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -107,7 +107,7 @@ private void bind(ConfigurationPropertiesBean bean) { * @since 2.2.0 */ public static void register(BeanDefinitionRegistry registry) { - Assert.notNull(registry, "Registry must not be null"); + Assert.notNull(registry, "'registry' must not be null"); if (!registry.containsBeanDefinition(BEAN_NAME)) { BeanDefinition definition = BeanDefinitionBuilder .rootBeanDefinition(ConfigurationPropertiesBindingPostProcessor.class) diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConversionServiceDeducer.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConversionServiceDeducer.java index 3f29066f03ce..4a74e54f2608 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConversionServiceDeducer.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/ConversionServiceDeducer.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,19 +19,13 @@ import java.util.ArrayList; import java.util.Collections; import java.util.List; +import java.util.Map; -import org.springframework.beans.factory.ListableBeanFactory; -import org.springframework.beans.factory.annotation.BeanFactoryAnnotationUtils; -import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; import org.springframework.boot.convert.ApplicationConversionService; import org.springframework.context.ApplicationContext; import org.springframework.context.ConfigurableApplicationContext; import org.springframework.core.convert.ConversionService; -import org.springframework.core.convert.converter.Converter; -import org.springframework.core.convert.converter.GenericConverter; import org.springframework.core.convert.support.DefaultConversionService; -import org.springframework.format.Formatter; -import org.springframework.format.FormatterRegistry; import org.springframework.format.support.FormattingConversionService; /** @@ -61,13 +55,9 @@ List getConversionServices() { private List getConversionServices(ConfigurableApplicationContext applicationContext) { List conversionServices = new ArrayList<>(); - ConverterBeans converterBeans = new ConverterBeans(applicationContext); + FormattingConversionService beansConverterService = new FormattingConversionService(); + Map converterBeans = addBeans(applicationContext, beansConverterService); if (!converterBeans.isEmpty()) { - FormattingConversionService beansConverterService = new FormattingConversionService(); - DefaultConversionService.addCollectionConverters(beansConverterService); - beansConverterService - .addConverter(new ConfigurationPropertiesCharSequenceToObjectConverter(beansConverterService)); - converterBeans.addTo(beansConverterService); conversionServices.add(beansConverterService); } if (applicationContext.getBeanFactory().getConversionService() != null) { @@ -83,50 +73,18 @@ private List getConversionServices(ConfigurableApplicationCon return conversionServices; } + private Map addBeans(ConfigurableApplicationContext applicationContext, + FormattingConversionService converterService) { + DefaultConversionService.addCollectionConverters(converterService); + converterService.addConverter(new ConfigurationPropertiesCharSequenceToObjectConverter(converterService)); + return ApplicationConversionService.addBeans(converterService, applicationContext.getBeanFactory(), + ConfigurationPropertiesBinding.VALUE); + } + private boolean hasUserDefinedConfigurationServiceBean() { String beanName = ConfigurableApplicationContext.CONVERSION_SERVICE_BEAN_NAME; return this.applicationContext.containsBean(beanName) && this.applicationContext.getAutowireCapableBeanFactory() .isTypeMatch(beanName, ConversionService.class); } - private static class ConverterBeans { - - @SuppressWarnings("rawtypes") - private final List converters; - - private final List genericConverters; - - @SuppressWarnings("rawtypes") - private final List formatters; - - ConverterBeans(ConfigurableApplicationContext applicationContext) { - ConfigurableListableBeanFactory beanFactory = applicationContext.getBeanFactory(); - this.converters = beans(Converter.class, ConfigurationPropertiesBinding.VALUE, beanFactory); - this.genericConverters = beans(GenericConverter.class, ConfigurationPropertiesBinding.VALUE, beanFactory); - this.formatters = beans(Formatter.class, ConfigurationPropertiesBinding.VALUE, beanFactory); - } - - private List beans(Class type, String qualifier, ListableBeanFactory beanFactory) { - return new ArrayList<>( - BeanFactoryAnnotationUtils.qualifiedBeansOfType(beanFactory, type, qualifier).values()); - } - - boolean isEmpty() { - return this.converters.isEmpty() && this.genericConverters.isEmpty() && this.formatters.isEmpty(); - } - - void addTo(FormatterRegistry registry) { - for (Converter converter : this.converters) { - registry.addConverter(converter); - } - for (GenericConverter genericConverter : this.genericConverters) { - registry.addConverter(genericConverter); - } - for (Formatter formatter : this.formatters) { - registry.addFormatter(formatter); - } - } - - } - } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/PropertyMapper.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/PropertyMapper.java index 87c03f77d3a6..9c5715394432 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/PropertyMapper.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/PropertyMapper.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -91,7 +91,7 @@ private Source whenNonNull(Source source) { * @return a new property mapper instance */ public PropertyMapper alwaysApplying(SourceOperator operator) { - Assert.notNull(operator, "Operator must not be null"); + Assert.notNull(operator, "'operator' must not be null"); return new PropertyMapper(this, operator); } @@ -104,7 +104,7 @@ public PropertyMapper alwaysApplying(SourceOperator operator) { * @see #from(Object) */ public Source from(Supplier supplier) { - Assert.notNull(supplier, "Supplier must not be null"); + Assert.notNull(supplier, "'supplier' must not be null"); Source source = getSource(supplier); if (this.sourceOperator != null) { source = this.sourceOperator.apply(source); @@ -167,7 +167,7 @@ public static final class Source { private final Predicate predicate; private Source(Supplier supplier, Predicate predicate) { - Assert.notNull(predicate, "Predicate must not be null"); + Assert.notNull(predicate, "'predicate' must not be null"); this.supplier = supplier; this.predicate = predicate; } @@ -190,7 +190,7 @@ public Source asInt(Function adapter) { * @return a new adapted source instance */ public Source as(Function adapter) { - Assert.notNull(adapter, "Adapter must not be null"); + Assert.notNull(adapter, "'adapter' must not be null"); Supplier test = () -> this.predicate.test(this.supplier.get()); Predicate predicate = (t) -> test.get(); Supplier supplier = () -> { @@ -266,7 +266,7 @@ public Source whenInstanceOf(Class target) { * @return a new filtered source instance */ public Source whenNot(Predicate predicate) { - Assert.notNull(predicate, "Predicate must not be null"); + Assert.notNull(predicate, "'predicate' must not be null"); return when(predicate.negate()); } @@ -277,7 +277,7 @@ public Source whenNot(Predicate predicate) { * @return a new filtered source instance */ public Source when(Predicate predicate) { - Assert.notNull(predicate, "Predicate must not be null"); + Assert.notNull(predicate, "'predicate' must not be null"); return new Source<>(this.supplier, (this.predicate != null) ? this.predicate.and(predicate) : predicate); } @@ -288,7 +288,7 @@ public Source when(Predicate predicate) { * filtered */ public void to(Consumer consumer) { - Assert.notNull(consumer, "Consumer must not be null"); + Assert.notNull(consumer, "'consumer' must not be null"); T value = this.supplier.get(); if (this.predicate.test(value)) { consumer.accept(value); @@ -307,8 +307,8 @@ public void to(Consumer consumer) { * @since 3.0.0 */ public R to(R instance, BiFunction mapper) { - Assert.notNull(instance, "Instance must not be null"); - Assert.notNull(mapper, "Mapper must not be null"); + Assert.notNull(instance, "'instance' must not be null"); + Assert.notNull(mapper, "'mapper' must not be null"); T value = this.supplier.get(); return (!this.predicate.test(value)) ? instance : mapper.apply(instance, value); } @@ -321,7 +321,7 @@ public R to(R instance, BiFunction mapper) { * @throws NoSuchElementException if the value has been filtered */ public R toInstance(Function factory) { - Assert.notNull(factory, "Factory must not be null"); + Assert.notNull(factory, "'factory' must not be null"); T value = this.supplier.get(); if (!this.predicate.test(value)) { throw new NoSuchElementException("No value present"); @@ -335,7 +335,7 @@ public R toInstance(Function factory) { * @param runnable the method to call if the value has not been filtered */ public void toCall(Runnable runnable) { - Assert.notNull(runnable, "Runnable must not be null"); + Assert.notNull(runnable, "'runnable' must not be null"); T value = this.supplier.get(); if (this.predicate.test(value)) { runnable.run(); diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/AbstractBindHandler.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/AbstractBindHandler.java index 7c0fd742fd8a..a0d102ff0983 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/AbstractBindHandler.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/AbstractBindHandler.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -42,7 +42,7 @@ public AbstractBindHandler() { * @param parent the parent handler */ public AbstractBindHandler(BindHandler parent) { - Assert.notNull(parent, "Parent must not be null"); + Assert.notNull(parent, "'parent' must not be null"); this.parent = parent; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BindResult.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BindResult.java index 2cc12866590a..d33010d79887 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BindResult.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BindResult.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -71,7 +71,7 @@ public boolean isBound() { * @param consumer block to execute if a value has been bound */ public void ifBound(Consumer consumer) { - Assert.notNull(consumer, "Consumer must not be null"); + Assert.notNull(consumer, "'consumer' must not be null"); if (this.value != null) { consumer.accept(this.value); } @@ -87,7 +87,7 @@ public void ifBound(Consumer consumer) { * to the value of this {@code BindResult}. */ public BindResult map(Function mapper) { - Assert.notNull(mapper, "Mapper must not be null"); + Assert.notNull(mapper, "'mapper' must not be null"); return of((this.value != null) ? mapper.apply(this.value) : null); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/Bindable.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/Bindable.java index 9e76a84e82c1..6b5d6f558e2e 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/Bindable.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/Bindable.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -192,7 +192,7 @@ public Bindable withAnnotations(Annotation... annotations) { public Bindable withExistingValue(T existingValue) { Assert.isTrue( existingValue == null || this.type.isArray() || this.boxedType.resolve().isInstance(existingValue), - () -> "ExistingValue must be an instance of " + this.type); + () -> "'existingValue' must be an instance of " + this.type); Assert.state(this.bindMethod != BindMethod.VALUE_OBJECT, () -> "An existing value cannot be provided when binding as a value object"); Supplier value = (existingValue != null) ? () -> existingValue : null; @@ -249,7 +249,7 @@ public Bindable withBindMethod(BindMethod bindMethod) { */ @SuppressWarnings("unchecked") public static Bindable ofInstance(T instance) { - Assert.notNull(instance, "Instance must not be null"); + Assert.notNull(instance, "'instance' must not be null"); Class type = (Class) instance.getClass(); return of(type).withExistingValue(instance); } @@ -262,7 +262,7 @@ public static Bindable ofInstance(T instance) { * @see #of(ResolvableType) */ public static Bindable of(Class type) { - Assert.notNull(type, "Type must not be null"); + Assert.notNull(type, "'type' must not be null"); return of(ResolvableType.forClass(type)); } @@ -306,7 +306,7 @@ public static Bindable> mapOf(Class keyType, Class valueT * @see #of(Class) */ public static Bindable of(ResolvableType type) { - Assert.notNull(type, "Type must not be null"); + Assert.notNull(type, "'type' must not be null"); ResolvableType boxedType = box(type); return new Bindable<>(type, boxedType, null, NO_ANNOTATIONS, NO_BIND_RESTRICTIONS, null); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BindableRuntimeHintsRegistrar.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BindableRuntimeHintsRegistrar.java index 6f70266024ad..bb4232282223 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BindableRuntimeHintsRegistrar.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BindableRuntimeHintsRegistrar.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -108,7 +108,7 @@ public void registerHints(RuntimeHints hints) { * @return a new {@link BindableRuntimeHintsRegistrar} instance */ public static BindableRuntimeHintsRegistrar forTypes(Iterable> types) { - Assert.notNull(types, "Types must not be null"); + Assert.notNull(types, "'types' must not be null"); return forTypes(StreamSupport.stream(types.spliterator(), false).toArray(Class[]::new)); } @@ -128,7 +128,7 @@ public static BindableRuntimeHintsRegistrar forTypes(Class... types) { * @since 3.0.8 */ public static BindableRuntimeHintsRegistrar forBindables(Iterable> bindables) { - Assert.notNull(bindables, "Bindables must not be null"); + Assert.notNull(bindables, "'bindables' must not be null"); return forBindables(StreamSupport.stream(bindables.spliterator(), false).toArray(Bindable[]::new)); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/Binder.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/Binder.java index 1630d2d8997a..b1b495d38801 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/Binder.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/Binder.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -25,7 +25,6 @@ import java.util.HashSet; import java.util.List; import java.util.Map; -import java.util.Objects; import java.util.Set; import java.util.function.Consumer; import java.util.function.Function; @@ -35,6 +34,7 @@ import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; import org.springframework.boot.context.properties.bind.Bindable.BindRestriction; import org.springframework.boot.context.properties.source.ConfigurationProperty; +import org.springframework.boot.context.properties.source.ConfigurationPropertyCaching; import org.springframework.boot.context.properties.source.ConfigurationPropertyName; import org.springframework.boot.context.properties.source.ConfigurationPropertySource; import org.springframework.boot.context.properties.source.ConfigurationPropertySources; @@ -45,6 +45,7 @@ import org.springframework.core.env.Environment; import org.springframework.format.support.DefaultFormattingConversionService; import org.springframework.util.Assert; +import org.springframework.util.ConcurrentReferenceHashMap; /** * A container object which Binds objects from one or more @@ -69,6 +70,10 @@ public class Binder { private final Map> dataObjectBinders; + private final Map cache = new ConcurrentReferenceHashMap<>(); + + private ConfigurationPropertyCaching configurationPropertyCaching; + /** * Create a new {@link Binder} instance for the specified sources. A * {@link DefaultFormattingConversionService} will be used for all conversion. @@ -184,11 +189,12 @@ public Binder(Iterable sources, PlaceholdersResolve public Binder(Iterable sources, PlaceholdersResolver placeholdersResolver, List conversionServices, Consumer propertyEditorInitializer, BindHandler defaultBindHandler, BindConstructorProvider constructorProvider) { - Assert.notNull(sources, "Sources must not be null"); + Assert.notNull(sources, "'sources' must not be null"); for (ConfigurationPropertySource source : sources) { - Assert.notNull(source, "Sources must not contain null elements"); + Assert.notNull(source, "'sources' must not contain null elements"); } this.sources = sources; + this.configurationPropertyCaching = ConfigurationPropertyCaching.get(sources); this.placeholdersResolver = (placeholdersResolver != null) ? placeholdersResolver : PlaceholdersResolver.NONE; this.bindConverter = BindConverter.get(conversionServices, propertyEditorInitializer); this.defaultBindHandler = (defaultBindHandler != null) ? defaultBindHandler : BindHandler.DEFAULT; @@ -332,8 +338,8 @@ public T bindOrCreate(ConfigurationPropertyName name, Bindable target, Bi } private T bind(ConfigurationPropertyName name, Bindable target, BindHandler handler, boolean create) { - Assert.notNull(name, "Name must not be null"); - Assert.notNull(target, "Target must not be null"); + Assert.notNull(name, "'name' must not be null"); + Assert.notNull(target, "'target' must not be null"); handler = (handler != null) ? handler : this.defaultBindHandler; Context context = new Context(); return bind(name, target, handler, context, false, create); @@ -341,17 +347,19 @@ private T bind(ConfigurationPropertyName name, Bindable target, BindHandl private T bind(ConfigurationPropertyName name, Bindable target, BindHandler handler, Context context, boolean allowRecursiveBinding, boolean create) { - try { - Bindable replacementTarget = handler.onStart(name, target, context); - if (replacementTarget == null) { - return handleBindResult(name, target, handler, context, null, create); + try (ConfigurationPropertyCaching.CacheOverride cacheOverride = this.configurationPropertyCaching.override()) { + try { + Bindable replacementTarget = handler.onStart(name, target, context); + if (replacementTarget == null) { + return handleBindResult(name, target, handler, context, null, create); + } + target = replacementTarget; + Object bound = bindObject(name, target, handler, context, allowRecursiveBinding); + return handleBindResult(name, target, handler, context, bound, create); + } + catch (Exception ex) { + return handleBindError(name, target, handler, context, ex); } - target = replacementTarget; - Object bound = bindObject(name, target, handler, context, allowRecursiveBinding); - return handleBindResult(name, target, handler, context, bound, create); - } - catch (Exception ex) { - return handleBindError(name, target, handler, context, ex); } } @@ -481,12 +489,13 @@ private Object bindDataObject(ConfigurationPropertyName name, Bindable target } private Object fromDataObjectBinders(BindMethod bindMethod, Function operation) { - return this.dataObjectBinders.get(bindMethod) - .stream() - .map(operation) - .filter(Objects::nonNull) - .findFirst() - .orElse(null); + for (DataObjectBinder dataObjectBinder : this.dataObjectBinders.get(bindMethod)) { + Object bound = operation.apply(dataObjectBinder); + if (bound != null) { + return bound; + } + } + return null; } private boolean isUnbindableBean(ConfigurationPropertyName name, Bindable target, Context context) { @@ -629,6 +638,10 @@ BindConverter getConverter() { return Binder.this.bindConverter; } + Map getCache() { + return Binder.this.cache; + } + @Override public Binder getBinder() { return Binder.this; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BoundPropertiesTrackingBindHandler.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BoundPropertiesTrackingBindHandler.java index f2a136be1d81..1b86b34d0ba2 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BoundPropertiesTrackingBindHandler.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/BoundPropertiesTrackingBindHandler.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,7 +33,7 @@ public class BoundPropertiesTrackingBindHandler extends AbstractBindHandler { private final Consumer consumer; public BoundPropertiesTrackingBindHandler(Consumer consumer) { - Assert.notNull(consumer, "Consumer must not be null"); + Assert.notNull(consumer, "'consumer' must not be null"); this.consumer = consumer; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/IndexedElementsBinder.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/IndexedElementsBinder.java index fa7704959dc3..1ff5f119cfa1 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/IndexedElementsBinder.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/IndexedElementsBinder.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,7 +43,13 @@ */ abstract class IndexedElementsBinder extends AggregateBinder { - private static final String INDEX_ZERO = "[0]"; + private static final String[] INDEXES; + static { + INDEXES = new String[10]; + for (int i = 0; i < INDEXES.length; i++) { + INDEXES[i] = "[" + i + "]"; + } + } IndexedElementsBinder(Context context) { super(context); @@ -100,15 +106,36 @@ private void bindValue(Bindable target, Collection collection, Resolv private void bindIndexed(ConfigurationPropertySource source, ConfigurationPropertyName root, AggregateElementBinder elementBinder, IndexedCollectionSupplier collection, ResolvableType elementType) { - MultiValueMap knownIndexedChildren = getKnownIndexedChildren(source, root); + int firstUnboundIndex = 0; + boolean hasBindingGap = false; for (int i = 0; i < Integer.MAX_VALUE; i++) { - ConfigurationPropertyName name = root.append((i != 0) ? "[" + i + "]" : INDEX_ZERO); + ConfigurationPropertyName name = appendIndex(root, i); Object value = elementBinder.bind(name, Bindable.of(elementType), source); - if (value == null) { + if (value != null) { + collection.get().add(value); + hasBindingGap = hasBindingGap || firstUnboundIndex > 0; + continue; + } + firstUnboundIndex = (firstUnboundIndex <= 0) ? i : firstUnboundIndex; + if (i - firstUnboundIndex > 10) { break; } + } + if (hasBindingGap) { + assertNoUnboundChildren(source, root, firstUnboundIndex); + } + } + + private ConfigurationPropertyName appendIndex(ConfigurationPropertyName root, int i) { + return root.append((i < INDEXES.length) ? INDEXES[i] : "[" + i + "]"); + } + + private void assertNoUnboundChildren(ConfigurationPropertySource source, ConfigurationPropertyName root, + int firstUnboundIndex) { + MultiValueMap knownIndexedChildren = getKnownIndexedChildren(source, root); + for (int i = 0; i < firstUnboundIndex; i++) { + ConfigurationPropertyName name = appendIndex(root, i); knownIndexedChildren.remove(name.getLastElement(Form.UNIFORM)); - collection.get().add(value); } assertNoUnboundChildren(source, knownIndexedChildren); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/JavaBeanBinder.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/JavaBeanBinder.java index 985a5fa995d0..af445bc2d9aa 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/JavaBeanBinder.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/JavaBeanBinder.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -28,6 +28,7 @@ import java.util.LinkedHashSet; import java.util.Map; import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; import java.util.function.BiConsumer; import java.util.function.Function; import java.util.function.Supplier; @@ -50,13 +51,16 @@ */ class JavaBeanBinder implements DataObjectBinder { + private static final String HAS_KNOWN_BINDABLE_PROPERTIES_CACHE = JavaBeanBinder.class.getName() + + ".HAS_KNOWN_BINDABLE_PROPERTIES_CACHE"; + static final JavaBeanBinder INSTANCE = new JavaBeanBinder(); @Override public T bind(ConfigurationPropertyName name, Bindable target, Context context, DataObjectPropertyBinder propertyBinder) { boolean hasKnownBindableProperties = target.getValue() != null && hasKnownBindableProperties(name, context); - Bean bean = Bean.get(target, hasKnownBindableProperties); + Bean bean = Bean.get(target, context, hasKnownBindableProperties); if (bean == null) { return null; } @@ -73,6 +77,16 @@ public T create(Bindable target, Context context) { } private boolean hasKnownBindableProperties(ConfigurationPropertyName name, Context context) { + Map cache = getHasKnownBindablePropertiesCache(context); + Boolean hasKnownBindableProperties = cache.get(name); + if (hasKnownBindableProperties == null) { + hasKnownBindableProperties = computeHasKnownBindableProperties(name, context); + cache.put(name, hasKnownBindableProperties); + } + return hasKnownBindableProperties; + } + + private boolean computeHasKnownBindableProperties(ConfigurationPropertyName name, Context context) { for (ConfigurationPropertySource source : context.getSources()) { if (source.containsDescendantOf(name) == ConfigurationPropertyState.PRESENT) { return true; @@ -81,6 +95,16 @@ private boolean hasKnownBindableProperties(ConfigurationPropertyName name, Conte return false; } + @SuppressWarnings("unchecked") + private Map getHasKnownBindablePropertiesCache(Context context) { + Object cache = context.getCache().get(HAS_KNOWN_BINDABLE_PROPERTIES_CACHE); + if (cache == null) { + cache = new ConcurrentHashMap(); + context.getCache().put(HAS_KNOWN_BINDABLE_PROPERTIES_CACHE, cache); + } + return (Map) cache; + } + private boolean bind(DataObjectPropertyBinder propertyBinder, Bean bean, BeanSupplier beanSupplier, Context context) { boolean bound = false; @@ -236,8 +260,6 @@ static BeanProperties of(Bindable bindable) { */ static class Bean extends BeanProperties { - private static Bean cached; - Bean(ResolvableType type, Class resolvedType) { super(type, resolvedType); } @@ -257,7 +279,7 @@ BeanSupplier getSupplier(Bindable target) { } @SuppressWarnings("unchecked") - static Bean get(Bindable bindable, boolean canCallGetValue) { + static Bean get(Bindable bindable, Context context, boolean canCallGetValue) { ResolvableType type = bindable.getType(); Class resolvedType = type.resolve(Object.class); Supplier value = bindable.getValue(); @@ -269,14 +291,26 @@ static Bean get(Bindable bindable, boolean canCallGetValue) { if (instance == null && !isInstantiable(resolvedType)) { return null; } - Bean bean = Bean.cached; - if (bean == null || !bean.isOfType(type, resolvedType)) { + Map> cache = getCache(context); + CacheKey cacheKey = new CacheKey(type, resolvedType); + Bean bean = cache.get(cacheKey); + if (bean == null) { bean = new Bean<>(type, resolvedType); - cached = bean; + cache.put(cacheKey, bean); } return (Bean) bean; } + @SuppressWarnings("unchecked") + private static Map> getCache(Context context) { + Map> cache = (Map>) context.getCache().get(Bean.class); + if (cache == null) { + cache = new ConcurrentHashMap<>(); + context.getCache().put(Bean.class, cache); + } + return cache; + } + private static boolean isInstantiable(Class type) { if (type.isInterface()) { return false; @@ -290,11 +324,8 @@ private static boolean isInstantiable(Class type) { } } - private boolean isOfType(ResolvableType type, Class resolvedType) { - if (getType().hasGenerics() || type.hasGenerics()) { - return getType().equals(type); - } - return getResolvedType() != null && getResolvedType().equals(resolvedType); + private record CacheKey(ResolvableType type, Class resolvedType) { + } } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/MapBinder.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/MapBinder.java index 573339edf929..62500e447d7e 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/MapBinder.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/MapBinder.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -155,44 +155,56 @@ private class EntryBinder { private final ResolvableType valueType; + private final Class resolvedValueType; + + private final boolean valueTreatedAsNestedMap; + + private final Bindable bindableMapType; + + private final Bindable bindableValueType; + EntryBinder(ConfigurationPropertyName root, Bindable target, AggregateElementBinder elementBinder) { this.root = root; this.elementBinder = elementBinder; this.mapType = target.getType().asMap(); this.keyType = this.mapType.getGeneric(0); this.valueType = this.mapType.getGeneric(1); + this.resolvedValueType = this.valueType.resolve(Object.class); + this.valueTreatedAsNestedMap = Object.class.equals(this.resolvedValueType); + this.bindableMapType = Bindable.of(this.mapType); + this.bindableValueType = Bindable.of(this.valueType); } void bindEntries(ConfigurationPropertySource source, Map map) { if (source instanceof IterableConfigurationPropertySource iterableSource) { for (ConfigurationPropertyName name : iterableSource) { - Bindable valueBindable = getValueBindable(name); ConfigurationPropertyName entryName = getEntryName(source, name); Object key = getContext().getConverter().convert(getKeyName(entryName), this.keyType); + Bindable valueBindable = getValueBindable(name); map.computeIfAbsent(key, (k) -> this.elementBinder.bind(entryName, valueBindable)); } } } private Bindable getValueBindable(ConfigurationPropertyName name) { - if (!this.root.isParentOf(name) && isValueTreatedAsNestedMap()) { - return Bindable.of(this.mapType); - } - return Bindable.of(this.valueType); + return (!isParentOf(name) && this.valueTreatedAsNestedMap) ? this.bindableMapType : this.bindableValueType; } private ConfigurationPropertyName getEntryName(ConfigurationPropertySource source, ConfigurationPropertyName name) { - Class resolved = this.valueType.resolve(Object.class); - if (Collection.class.isAssignableFrom(resolved) || this.valueType.isArray()) { + if (Collection.class.isAssignableFrom(this.resolvedValueType) || this.valueType.isArray()) { return chopNameAtNumericIndex(name); } - if (!this.root.isParentOf(name) && (isValueTreatedAsNestedMap() || !isScalarValue(source, name))) { + if (!isParentOf(name) && (this.valueTreatedAsNestedMap || !isScalarValue(source, name))) { return name.chop(this.root.getNumberOfElements() + 1); } return name; } + private boolean isParentOf(ConfigurationPropertyName name) { + return this.root.isParentOf(name); + } + private ConfigurationPropertyName chopNameAtNumericIndex(ConfigurationPropertyName name) { int start = this.root.getNumberOfElements() + 1; int size = name.getNumberOfElements(); @@ -204,10 +216,6 @@ private ConfigurationPropertyName chopNameAtNumericIndex(ConfigurationPropertyNa return name; } - private boolean isValueTreatedAsNestedMap() { - return Object.class.equals(this.valueType.resolve(Object.class)); - } - private boolean isScalarValue(ConfigurationPropertySource source, ConfigurationPropertyName name) { Class resolved = this.valueType.resolve(Object.class); if (!resolved.getName().startsWith("java.lang") && !resolved.isEnum()) { diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/PropertySourcesPlaceholdersResolver.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/PropertySourcesPlaceholdersResolver.java index 67e432fab521..7bf6807ef3d2 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/PropertySourcesPlaceholdersResolver.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/PropertySourcesPlaceholdersResolver.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -74,9 +74,9 @@ protected String resolvePlaceholder(String placeholder) { } private static PropertySources getSources(Environment environment) { - Assert.notNull(environment, "Environment must not be null"); + Assert.notNull(environment, "'environment' must not be null"); Assert.isInstanceOf(ConfigurableEnvironment.class, environment, - "Environment must be a ConfigurableEnvironment"); + "'environment' must be a ConfigurableEnvironment"); return ((ConfigurableEnvironment) environment).getPropertySources(); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/ValueObjectBinder.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/ValueObjectBinder.java index d304bff3baac..9ae5dfece55c 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/ValueObjectBinder.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/ValueObjectBinder.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -29,6 +29,7 @@ import java.util.List; import java.util.Map; import java.util.Optional; +import java.util.concurrent.ConcurrentHashMap; import java.util.function.Consumer; import kotlin.reflect.KFunction; @@ -73,7 +74,7 @@ class ValueObjectBinder implements DataObjectBinder { @Override public T bind(ConfigurationPropertyName name, Bindable target, Binder.Context context, DataObjectPropertyBinder propertyBinder) { - ValueObject valueObject = ValueObject.get(target, this.constructorProvider, context, Discoverer.LENIENT); + ValueObject valueObject = ValueObject.get(target, context, this.constructorProvider, Discoverer.LENIENT); if (valueObject == null) { return null; } @@ -94,7 +95,7 @@ public T bind(ConfigurationPropertyName name, Bindable target, Binder.Con @Override public T create(Bindable target, Binder.Context context) { - ValueObject valueObject = ValueObject.get(target, this.constructorProvider, context, Discoverer.LENIENT); + ValueObject valueObject = ValueObject.get(target, context, this.constructorProvider, Discoverer.LENIENT); if (valueObject == null) { return null; } @@ -109,7 +110,7 @@ public T create(Bindable target, Binder.Context context) { @Override public void onUnableToCreateInstance(Bindable target, Context context, RuntimeException exception) { try { - ValueObject.get(target, this.constructorProvider, context, Discoverer.STRICT); + ValueObject.get(target, context, this.constructorProvider, Discoverer.STRICT); } catch (Exception ex) { exception.addSuppressed(ex); @@ -191,6 +192,8 @@ private boolean isAggregate(Class type) { */ private abstract static class ValueObject { + private static final Object NONE = new Object(); + private final Constructor constructor; protected ValueObject(Constructor constructor) { @@ -204,24 +207,53 @@ T instantiate(List args) { abstract List getConstructorParameters(); @SuppressWarnings("unchecked") - static ValueObject get(Bindable bindable, BindConstructorProvider constructorProvider, - Binder.Context context, ParameterNameDiscoverer parameterNameDiscoverer) { - Class type = (Class) bindable.getType().resolve(); - if (type == null || type.isEnum() || Modifier.isAbstract(type.getModifiers())) { + static ValueObject get(Bindable bindable, Binder.Context context, + BindConstructorProvider constructorProvider, ParameterNameDiscoverer parameterNameDiscoverer) { + Class resolvedType = (Class) bindable.getType().resolve(); + if (resolvedType == null || resolvedType.isEnum() || Modifier.isAbstract(resolvedType.getModifiers())) { return null; } + Map cache = getCache(context); + CacheKey cacheKey = new CacheKey(bindable, constructorProvider, parameterNameDiscoverer); + Object valueObject = cache.get(cacheKey); + if (valueObject == null) { + valueObject = get(bindable, context, constructorProvider, parameterNameDiscoverer, resolvedType); + cache.put(cacheKey, (valueObject != null) ? valueObject : NONE); + } + return (valueObject != NONE) ? (ValueObject) valueObject : null; + } + + @SuppressWarnings("unchecked") + private static ValueObject get(Bindable bindable, Binder.Context context, + BindConstructorProvider constructorProvider, ParameterNameDiscoverer parameterNameDiscoverer, + Class resolvedType) { Constructor bindConstructor = constructorProvider.getBindConstructor(bindable, context.isNestedConstructorBinding()); if (bindConstructor == null) { return null; } - if (KotlinDetector.isKotlinType(type)) { + if (KotlinDetector.isKotlinType(resolvedType)) { return KotlinValueObject.get((Constructor) bindConstructor, bindable.getType(), parameterNameDiscoverer); } return DefaultValueObject.get(bindConstructor, bindable.getType(), parameterNameDiscoverer); } + @SuppressWarnings("unchecked") + private static Map getCache(Context context) { + Map cache = (Map) context.getCache().get(ValueObject.class); + if (cache == null) { + cache = new ConcurrentHashMap<>(); + context.getCache().put(ValueObject.class, cache); + } + return cache; + } + + private record CacheKey(Bindable bindable, BindConstructorProvider constructorProvider, + ParameterNameDiscoverer parameterNameDiscoverer) { + + } + } /** diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/validation/BindValidationException.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/validation/BindValidationException.java index b1be6d24c85d..d674cc08c257 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/validation/BindValidationException.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/validation/BindValidationException.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -33,7 +33,7 @@ public class BindValidationException extends RuntimeException { BindValidationException(ValidationErrors validationErrors) { super(getMessage(validationErrors)); - Assert.notNull(validationErrors, "ValidationErrors must not be null"); + Assert.notNull(validationErrors, "'validationErrors' must not be null"); this.validationErrors = validationErrors; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/validation/ValidationErrors.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/validation/ValidationErrors.java index 1dd1d5784655..390894989fbe 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/validation/ValidationErrors.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/bind/validation/ValidationErrors.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2022 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -49,9 +49,9 @@ public class ValidationErrors implements Iterable { ValidationErrors(ConfigurationPropertyName name, Set boundProperties, List errors) { - Assert.notNull(name, "Name must not be null"); - Assert.notNull(boundProperties, "BoundProperties must not be null"); - Assert.notNull(errors, "Errors must not be null"); + Assert.notNull(name, "'name' must not be null"); + Assert.notNull(boundProperties, "'boundProperties' must not be null"); + Assert.notNull(errors, "'errors' must not be null"); this.name = name; this.boundProperties = Collections.unmodifiableSet(boundProperties); this.errors = convertErrors(name, boundProperties, errors); diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/AliasedConfigurationPropertySource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/AliasedConfigurationPropertySource.java index 6c48ed3b43f0..33e89baab2cb 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/AliasedConfigurationPropertySource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/AliasedConfigurationPropertySource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,15 +31,15 @@ class AliasedConfigurationPropertySource implements ConfigurationPropertySource private final ConfigurationPropertyNameAliases aliases; AliasedConfigurationPropertySource(ConfigurationPropertySource source, ConfigurationPropertyNameAliases aliases) { - Assert.notNull(source, "Source must not be null"); - Assert.notNull(aliases, "Aliases must not be null"); + Assert.notNull(source, "'source' must not be null"); + Assert.notNull(aliases, "'aliases' must not be null"); this.source = source; this.aliases = aliases; } @Override public ConfigurationProperty getConfigurationProperty(ConfigurationPropertyName name) { - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); ConfigurationProperty result = getSource().getConfigurationProperty(name); if (result == null) { ConfigurationPropertyName aliasedName = getAliases().getNameForAlias(name); @@ -50,7 +50,7 @@ public ConfigurationProperty getConfigurationProperty(ConfigurationPropertyName @Override public ConfigurationPropertyState containsDescendantOf(ConfigurationPropertyName name) { - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); ConfigurationPropertyState result = this.source.containsDescendantOf(name); if (result != ConfigurationPropertyState.ABSENT) { return result; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationProperty.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationProperty.java index 27f4948e1a12..4fa1d275bef8 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationProperty.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationProperty.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,8 +48,8 @@ public ConfigurationProperty(ConfigurationPropertyName name, Object value, Origi private ConfigurationProperty(ConfigurationPropertySource source, ConfigurationPropertyName name, Object value, Origin origin) { - Assert.notNull(name, "Name must not be null"); - Assert.notNull(value, "Value must not be null"); + Assert.notNull(name, "'name' must not be null"); + Assert.notNull(value, "'value' must not be null"); this.source = source; this.name = name; this.value = value; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyCaching.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyCaching.java index 4471b9dea475..b24e5218423a 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyCaching.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyCaching.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -51,6 +51,14 @@ public interface ConfigurationPropertyCaching { */ void clear(); + /** + * Override caching to temporarily enable it. Once caching is no longer needed the + * returned {@link CacheOverride} should be closed to restore previous cache settings. + * @return a {@link CacheOverride} + * @since 3.5.0 + */ + CacheOverride override(); + /** * Get for all configuration property sources in the environment. * @param environment the spring environment @@ -92,7 +100,7 @@ static ConfigurationPropertyCaching get(Iterable so * @return a caching instance that controls the matching source */ static ConfigurationPropertyCaching get(Iterable sources, Object underlyingSource) { - Assert.notNull(sources, "Sources must not be null"); + Assert.notNull(sources, "'sources' must not be null"); if (underlyingSource == null) { return new ConfigurationPropertySourcesCaching(sources); } @@ -107,4 +115,17 @@ static ConfigurationPropertyCaching get(Iterable so throw new IllegalStateException("Unable to find cache from configuration property sources"); } + /** + * {@link AutoCloseable} used to control a + * {@link ConfigurationPropertyCaching#override() cache override}. + * + * @since 3.5.0 + */ + interface CacheOverride extends AutoCloseable { + + @Override + void close(); + + } + } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyName.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyName.java index bd1e7f1dda00..87d237bc2368 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyName.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyName.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,8 +20,10 @@ import java.util.Collection; import java.util.Collections; import java.util.List; +import java.util.Locale; import java.util.Map; import java.util.function.Function; +import java.util.function.IntFunction; import org.springframework.util.Assert; import org.springframework.util.StringUtils; @@ -63,10 +65,14 @@ public final class ConfigurationPropertyName implements Comparable= name.getNumberOfElements()) { return false; } - return elementsEqual(name); + return endsWithElementsEqualTo(name); } @Override @@ -357,10 +363,20 @@ public boolean equals(Object obj) { && other.elements.canShortcutWithSource(ElementType.UNIFORM)) { return toString().equals(other.toString()); } - return elementsEqual(other); + if (hashCode() != other.hashCode()) { + return false; + } + if (toStringMatches(toString(), other.toString())) { + return true; + } + return endsWithElementsEqualTo(other); + } + + private boolean toStringMatches(String s1, String s2) { + return s1.hashCode() == s2.hashCode() && s1.equals(s2); } - private boolean elementsEqual(ConfigurationPropertyName name) { + private boolean endsWithElementsEqualTo(ConfigurationPropertyName name) { for (int i = this.elements.getSize() - 1; i >= 0; i--) { if (elementDiffers(this.elements, name.elements, i)) { return false; @@ -508,34 +524,48 @@ public int hashCode() { Elements elements = this.elements; if (hashCode == 0 && elements.getSize() != 0) { for (int elementIndex = 0; elementIndex < elements.getSize(); elementIndex++) { - int elementHashCode = 0; - boolean indexed = elements.getType(elementIndex).isIndexed(); - int length = elements.getLength(elementIndex); - for (int i = 0; i < length; i++) { - char ch = elements.charAt(elementIndex, i); - if (!indexed) { - ch = Character.toLowerCase(ch); - } - if (ElementsParser.isAlphaNumeric(ch)) { - elementHashCode = 31 * elementHashCode + ch; - } - } - hashCode = 31 * hashCode + elementHashCode; + hashCode = 31 * hashCode + elements.hashCode(elementIndex); } this.hashCode = hashCode; } return hashCode; } + ConfigurationPropertyName asSystemEnvironmentLegacyName() { + ConfigurationPropertyName name = this.systemEnvironmentLegacyName; + if (name == null) { + name = ConfigurationPropertyName + .ofIfValid(buildSimpleToString('.', (i) -> getElement(i, Form.DASHED).replace('-', '.'))); + this.systemEnvironmentLegacyName = (name != null) ? name : EMPTY; + } + return (name != EMPTY) ? name : null; + } + @Override public String toString() { - if (this.string == null) { - this.string = buildToString(); + return toString(ToStringFormat.DEFAULT); + } + + String toString(ToStringFormat format) { + String string = this.string[format.ordinal()]; + if (string == null) { + string = buildToString(format); + this.string[format.ordinal()] = string; } - return this.string; + return string; } - private String buildToString() { + private String buildToString(ToStringFormat format) { + return switch (format) { + case DEFAULT -> buildDefaultToString(); + case SYSTEM_ENVIRONMENT -> + buildSimpleToString('_', (i) -> getElement(i, Form.UNIFORM).toUpperCase(Locale.ENGLISH)); + case LEGACY_SYSTEM_ENVIRONMENT -> buildSimpleToString('_', + (i) -> getElement(i, Form.ORIGINAL).replace('-', '_').toUpperCase(Locale.ENGLISH)); + }; + } + + private String buildDefaultToString() { if (this.elements.canShortcutWithSource(ElementType.UNIFORM, ElementType.DASHED)) { return this.elements.getSource().toString(); } @@ -558,6 +588,32 @@ private String buildToString() { return result.toString(); } + private String buildSimpleToString(char joinChar, IntFunction elementConverter) { + StringBuilder result = new StringBuilder(); + for (int i = 0; i < getNumberOfElements(); i++) { + if (!result.isEmpty()) { + result.append(joinChar); + } + result.append(elementConverter.apply(i)); + } + return result.toString(); + } + + boolean hasDashedElement() { + Boolean hasDashedElement = this.hasDashedElement; + if (hasDashedElement != null) { + return hasDashedElement; + } + for (int i = 0; i < getNumberOfElements(); i++) { + if (getElement(i, Form.DASHED).indexOf('-') != -1) { + this.hasDashedElement = true; + return true; + } + } + this.hasDashedElement = false; + return false; + } + /** * Returns if the given name is valid. If this method returns {@code true} then the * name may be used with {@link #of(CharSequence)} without throwing an exception. @@ -612,7 +668,7 @@ private static Elements elementsOf(CharSequence name, boolean returnNullIfInvali private static Elements elementsOf(CharSequence name, boolean returnNullIfInvalid, int parserCapacity) { if (name == null) { - Assert.isTrue(returnNullIfInvalid, "Name must not be null"); + Assert.isTrue(returnNullIfInvalid, "'name' must not be null"); return null; } if (name.isEmpty()) { @@ -737,7 +793,7 @@ private static class Elements { private static final ElementType[] NO_TYPE = {}; - public static final Elements EMPTY = new Elements("", 0, NO_POSITION, NO_POSITION, NO_TYPE, null); + public static final Elements EMPTY = new Elements("", 0, NO_POSITION, NO_POSITION, NO_TYPE, null, null); private final CharSequence source; @@ -749,6 +805,8 @@ private static class Elements { private final ElementType[] type; + private final int[] hashCode; + /** * Contains any resolved elements or can be {@code null} if there aren't any. * Resolved elements allow us to modify the element values in some way (or example @@ -759,30 +817,35 @@ private static class Elements { */ private final CharSequence[] resolved; - Elements(CharSequence source, int size, int[] start, int[] end, ElementType[] type, CharSequence[] resolved) { + Elements(CharSequence source, int size, int[] start, int[] end, ElementType[] type, int[] hashCode, + CharSequence[] resolved) { this.source = source; this.size = size; this.start = start; this.end = end; this.type = type; + this.hashCode = (hashCode != null) ? hashCode : new int[size]; this.resolved = resolved; } Elements append(Elements additional) { int size = this.size + additional.size; ElementType[] type = new ElementType[size]; + int[] hashCode = new int[size]; System.arraycopy(this.type, 0, type, 0, this.size); System.arraycopy(additional.type, 0, type, this.size, additional.size); + System.arraycopy(this.hashCode, 0, hashCode, 0, this.size); + System.arraycopy(additional.hashCode, 0, hashCode, this.size, additional.size); CharSequence[] resolved = newResolved(0, size); for (int i = 0; i < additional.size; i++) { resolved[this.size + i] = additional.get(i); } - return new Elements(this.source, size, this.start, this.end, type, resolved); + return new Elements(this.source, size, this.start, this.end, type, hashCode, resolved); } Elements chop(int size) { CharSequence[] resolved = newResolved(0, size); - return new Elements(this.source, size, this.start, this.end, this.type, resolved); + return new Elements(this.source, size, this.start, this.end, this.type, this.hashCode, resolved); } Elements subElements(int offset) { @@ -794,7 +857,9 @@ Elements subElements(int offset) { System.arraycopy(this.end, offset, end, 0, size); ElementType[] type = new ElementType[size]; System.arraycopy(this.type, offset, type, 0, size); - return new Elements(this.source, size, start, end, type, resolved); + int[] hashCode = new int[size]; + System.arraycopy(this.hashCode, offset, hashCode, 0, size); + return new Elements(this.source, size, start, end, type, hashCode, resolved); } private CharSequence[] newResolved(int offset, int size) { @@ -839,6 +904,25 @@ ElementType getType(int index) { return this.type[index]; } + int hashCode(int index) { + int hashCode = this.hashCode[index]; + if (hashCode == 0) { + boolean indexed = getType(index).isIndexed(); + int length = getLength(index); + for (int i = 0; i < length; i++) { + char ch = charAt(index, i); + if (!indexed) { + ch = Character.toLowerCase(ch); + } + if (ElementsParser.isAlphaNumeric(ch)) { + hashCode = 31 * hashCode + ch; + } + } + this.hashCode[index] = hashCode; + } + return hashCode; + } + CharSequence getSource() { return this.source; } @@ -951,7 +1035,7 @@ else if (!type.isIndexed() && ch == this.separator) { type = ElementType.NON_UNIFORM; } add(start, length, type, valueProcessor); - return new Elements(this.source, this.size, this.start, this.end, this.type, this.resolved); + return new Elements(this.source, this.size, this.start, this.end, this.type, null, this.resolved); } private ElementType updateType(ElementType existingType, char ch, int index) { @@ -1106,4 +1190,13 @@ private interface ElementCharPredicate { } + /** + * Formats for {@code toString}. + */ + enum ToStringFormat { + + DEFAULT, SYSTEM_ENVIRONMENT, LEGACY_SYSTEM_ENVIRONMENT + + } + } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyNameAliases.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyNameAliases.java index 5a077ad00a19..d44ab92475ff 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyNameAliases.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyNameAliases.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -50,15 +50,15 @@ public ConfigurationPropertyNameAliases(ConfigurationPropertyName name, Configur } public void addAliases(String name, String... aliases) { - Assert.notNull(name, "Name must not be null"); - Assert.notNull(aliases, "Aliases must not be null"); + Assert.notNull(name, "'name' must not be null"); + Assert.notNull(aliases, "'aliases' must not be null"); addAliases(ConfigurationPropertyName.of(name), Arrays.stream(aliases).map(ConfigurationPropertyName::of).toArray(ConfigurationPropertyName[]::new)); } public void addAliases(ConfigurationPropertyName name, ConfigurationPropertyName... aliases) { - Assert.notNull(name, "Name must not be null"); - Assert.notNull(aliases, "Aliases must not be null"); + Assert.notNull(name, "'name' must not be null"); + Assert.notNull(aliases, "'aliases' must not be null"); this.aliases.addAll(name, Arrays.asList(aliases)); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertySourcesCaching.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertySourcesCaching.java index 60df84baecf2..dd8785ddb11e 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertySourcesCaching.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertySourcesCaching.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,6 +17,8 @@ package org.springframework.boot.context.properties.source; import java.time.Duration; +import java.util.ArrayList; +import java.util.List; import java.util.function.Consumer; /** @@ -53,6 +55,13 @@ public void clear() { forEach(ConfigurationPropertyCaching::clear); } + @Override + public CacheOverride override() { + CacheOverrides override = new CacheOverrides(); + forEach(override::add); + return override; + } + private void forEach(Consumer action) { if (this.sources != null) { for (ConfigurationPropertySource source : this.sources) { @@ -64,4 +73,22 @@ private void forEach(Consumer action) { } } + /** + * Composite {@link CacheOverride}. + */ + private final class CacheOverrides implements CacheOverride { + + private List overrides = new ArrayList<>(); + + void add(ConfigurationPropertyCaching caching) { + this.overrides.add(caching.override()); + } + + @Override + public void close() { + this.overrides.forEach(CacheOverride::close); + } + + } + } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyState.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyState.java index 1b88a9a6e1ff..4bb7b2c5241f 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyState.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/ConfigurationPropertyState.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,8 +56,8 @@ public enum ConfigurationPropertyState { * {@link #ABSENT}. */ static ConfigurationPropertyState search(Iterable source, Predicate predicate) { - Assert.notNull(source, "Source must not be null"); - Assert.notNull(predicate, "Predicate must not be null"); + Assert.notNull(source, "'source' must not be null"); + Assert.notNull(predicate, "'predicate' must not be null"); for (T item : source) { if (predicate.test(item)) { return PRESENT; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/FilteredConfigurationPropertiesSource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/FilteredConfigurationPropertiesSource.java index 86a266f01335..92f72918f97f 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/FilteredConfigurationPropertiesSource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/FilteredConfigurationPropertiesSource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -34,8 +34,8 @@ class FilteredConfigurationPropertiesSource implements ConfigurationPropertySour FilteredConfigurationPropertiesSource(ConfigurationPropertySource source, Predicate filter) { - Assert.notNull(source, "Source must not be null"); - Assert.notNull(filter, "Filter must not be null"); + Assert.notNull(source, "'source' must not be null"); + Assert.notNull(filter, "'filter' must not be null"); this.source = source; this.filter = filter; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/InvalidConfigurationPropertyValueException.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/InvalidConfigurationPropertyValueException.java index 1bef986b34ef..60470e3798a0 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/InvalidConfigurationPropertyValueException.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/InvalidConfigurationPropertyValueException.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -48,7 +48,7 @@ public InvalidConfigurationPropertyValueException(String name, Object value, Str InvalidConfigurationPropertyValueException(String name, Object value, String reason, Throwable cause) { super("Property " + name + " with value '" + value + "' is invalid: " + reason, cause); - Assert.notNull(name, "Name must not be null"); + Assert.notNull(name, "'name' must not be null"); this.name = name; this.value = value; this.reason = reason; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/MapConfigurationPropertySource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/MapConfigurationPropertySource.java index b24f13fa94fa..87fa56a90b13 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/MapConfigurationPropertySource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/MapConfigurationPropertySource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2020 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -56,7 +56,7 @@ public MapConfigurationPropertySource() { public MapConfigurationPropertySource(Map map) { this.source = new LinkedHashMap<>(); MapPropertySource mapPropertySource = new MapPropertySource("source", this.source); - this.delegate = new SpringIterableConfigurationPropertySource(mapPropertySource, DEFAULT_MAPPERS); + this.delegate = new SpringIterableConfigurationPropertySource(mapPropertySource, false, DEFAULT_MAPPERS); putAll(map); } @@ -65,7 +65,7 @@ public MapConfigurationPropertySource(Map map) { * @param map the source map */ public void putAll(Map map) { - Assert.notNull(map, "Map must not be null"); + Assert.notNull(map, "'map' must not be null"); assertNotReadOnlySystemAttributesMap(map); map.forEach(this::put); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/MutuallyExclusiveConfigurationPropertiesException.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/MutuallyExclusiveConfigurationPropertiesException.java index b0fa0a3e8167..cc868e6759bb 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/MutuallyExclusiveConfigurationPropertiesException.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/MutuallyExclusiveConfigurationPropertiesException.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -84,9 +84,9 @@ private static Set asSet(Collection collection) { private static String buildMessage(Set mutuallyExclusiveNames, Set configuredNames) { Assert.isTrue(configuredNames != null && configuredNames.size() > 1, - "ConfiguredNames must contain 2 or more names"); + "'configuredNames' must contain 2 or more names"); Assert.isTrue(mutuallyExclusiveNames != null && mutuallyExclusiveNames.size() > 1, - "MutuallyExclusiveNames must contain 2 or more names"); + "'mutuallyExclusiveNames' must contain 2 or more names"); return "The configuration properties '" + String.join(", ", mutuallyExclusiveNames) + "' are mutually exclusive and '" + String.join(", ", configuredNames) + "' have been configured together"; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/PrefixedConfigurationPropertySource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/PrefixedConfigurationPropertySource.java index 7e2066258240..34192c1216ec 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/PrefixedConfigurationPropertySource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/PrefixedConfigurationPropertySource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -30,8 +30,8 @@ class PrefixedConfigurationPropertySource implements ConfigurationPropertySource private final ConfigurationPropertyName prefix; PrefixedConfigurationPropertySource(ConfigurationPropertySource source, String prefix) { - Assert.notNull(source, "Source must not be null"); - Assert.hasText(prefix, "Prefix must not be empty"); + Assert.notNull(source, "'source' must not be null"); + Assert.hasText(prefix, "'prefix' must not be empty"); this.source = source; this.prefix = ConfigurationPropertyName.of(prefix); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SoftReferenceConfigurationPropertyCache.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SoftReferenceConfigurationPropertyCache.java index d71b35e842fd..5d596709e5df 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SoftReferenceConfigurationPropertyCache.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SoftReferenceConfigurationPropertyCache.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2021 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,7 @@ import java.lang.ref.SoftReference; import java.time.Duration; import java.time.Instant; +import java.util.concurrent.atomic.AtomicBoolean; import java.util.function.Supplier; import java.util.function.UnaryOperator; @@ -33,6 +34,9 @@ class SoftReferenceConfigurationPropertyCache implements ConfigurationPropert private static final Duration UNLIMITED = Duration.ZERO; + static final CacheOverride NO_OP_OVERRIDE = () -> { + }; + private final boolean neverExpire; private volatile Duration timeToLive; @@ -65,6 +69,25 @@ public void clear() { this.lastAccessed = null; } + @Override + public CacheOverride override() { + if (this.neverExpire) { + return NO_OP_OVERRIDE; + } + ActiveCacheOverride override = new ActiveCacheOverride(this); + if (override.timeToLive() == null) { + // Ensure we don't use stale data on the first access + clear(); + } + this.timeToLive = UNLIMITED; + return override; + } + + void restore(ActiveCacheOverride override) { + this.timeToLive = override.timeToLive(); + this.lastAccessed = override.lastAccessed(); + } + /** * Get a value from the cache, creating it if necessary. * @param factory a factory used to create the item if there is no reference to it. @@ -111,4 +134,23 @@ protected void setValue(T value) { this.value = new SoftReference<>(value); } + /** + * An active {@link CacheOverride} with a stored time-to-live. + */ + private record ActiveCacheOverride(SoftReferenceConfigurationPropertyCache cache, Duration timeToLive, + Instant lastAccessed, AtomicBoolean active) implements CacheOverride { + + ActiveCacheOverride(SoftReferenceConfigurationPropertyCache cache) { + this(cache, cache.timeToLive, cache.lastAccessed, new AtomicBoolean()); + } + + @Override + public void close() { + if (active().compareAndSet(false, true)) { + this.cache.restore(this); + } + } + + } + } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringConfigurationPropertySource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringConfigurationPropertySource.java index 965b2ed14ab9..7065a72a9597 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringConfigurationPropertySource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringConfigurationPropertySource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -59,17 +59,22 @@ class SpringConfigurationPropertySource implements ConfigurationPropertySource { private final PropertySource propertySource; + private final boolean systemEnvironmentSource; + private final PropertyMapper[] mappers; /** * Create a new {@link SpringConfigurationPropertySource} implementation. * @param propertySource the source property source + * @param systemEnvironmentSource if the source is from the system environment * @param mappers the property mappers */ - SpringConfigurationPropertySource(PropertySource propertySource, PropertyMapper... mappers) { - Assert.notNull(propertySource, "PropertySource must not be null"); - Assert.isTrue(mappers.length > 0, "Mappers must contain at least one item"); + SpringConfigurationPropertySource(PropertySource propertySource, boolean systemEnvironmentSource, + PropertyMapper... mappers) { + Assert.notNull(propertySource, "'propertySource' must not be null"); + Assert.isTrue(mappers.length > 0, "'mappers' must contain at least one item"); this.propertySource = propertySource; + this.systemEnvironmentSource = systemEnvironmentSource; this.mappers = mappers; } @@ -81,7 +86,7 @@ public ConfigurationProperty getConfigurationProperty(ConfigurationPropertyName for (PropertyMapper mapper : this.mappers) { try { for (String candidate : mapper.map(name)) { - Object value = getPropertySource().getProperty(candidate); + Object value = getPropertySourceProperty(candidate); if (value != null) { Origin origin = PropertySourceOrigin.get(this.propertySource, candidate); return ConfigurationProperty.of(this, name, value, origin); @@ -95,6 +100,18 @@ public ConfigurationProperty getConfigurationProperty(ConfigurationPropertyName return null; } + protected final Object getPropertySourceProperty(String name) { + // Save calls to SystemEnvironmentPropertySource.resolvePropertyName(...) + // since we've already done the mapping + PropertySource propertySource = getPropertySource(); + return (!this.systemEnvironmentSource) ? propertySource.getProperty(name) + : getSystemEnvironmentProperty(((SystemEnvironmentPropertySource) propertySource).getSource(), name); + } + + Object getSystemEnvironmentProperty(Map systemEnvironment, String name) { + return systemEnvironment.get(name); + } + @Override public ConfigurationPropertyState containsDescendantOf(ConfigurationPropertyName name) { PropertySource source = getPropertySource(); @@ -127,6 +144,10 @@ protected PropertySource getPropertySource() { return this.propertySource; } + protected final boolean isSystemEnvironmentSource() { + return this.systemEnvironmentSource; + } + protected final PropertyMapper[] getMappers() { return this.mappers; } @@ -144,25 +165,20 @@ public String toString() { * {@link SpringIterableConfigurationPropertySource} instance */ static SpringConfigurationPropertySource from(PropertySource source) { - Assert.notNull(source, "Source must not be null"); - PropertyMapper[] mappers = getPropertyMappers(source); - if (isFullEnumerable(source)) { - return new SpringIterableConfigurationPropertySource((EnumerablePropertySource) source, mappers); - } - return new SpringConfigurationPropertySource(source, mappers); - } - - private static PropertyMapper[] getPropertyMappers(PropertySource source) { - if (source instanceof SystemEnvironmentPropertySource && hasSystemEnvironmentName(source)) { - return SYSTEM_ENVIRONMENT_MAPPERS; - } - return DEFAULT_MAPPERS; + Assert.notNull(source, "'source' must not be null"); + boolean systemEnvironmentSource = isSystemEnvironmentPropertySource(source); + PropertyMapper[] mappers = (!systemEnvironmentSource) ? DEFAULT_MAPPERS : SYSTEM_ENVIRONMENT_MAPPERS; + return (!isFullEnumerable(source)) + ? new SpringConfigurationPropertySource(source, systemEnvironmentSource, mappers) + : new SpringIterableConfigurationPropertySource((EnumerablePropertySource) source, + systemEnvironmentSource, mappers); } - private static boolean hasSystemEnvironmentName(PropertySource source) { + private static boolean isSystemEnvironmentPropertySource(PropertySource source) { String name = source.getName(); - return StandardEnvironment.SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME.equals(name) - || name.endsWith("-" + StandardEnvironment.SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME); + return (source instanceof SystemEnvironmentPropertySource) + && (StandardEnvironment.SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME.equals(name) + || name.endsWith("-" + StandardEnvironment.SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME)); } private static boolean isFullEnumerable(PropertySource source) { diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringConfigurationPropertySources.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringConfigurationPropertySources.java index b3d608d935ed..4fe4a5625f03 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringConfigurationPropertySources.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringConfigurationPropertySources.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -46,7 +46,7 @@ class SpringConfigurationPropertySources implements Iterable> sources) { - Assert.notNull(sources, "Sources must not be null"); + Assert.notNull(sources, "'sources' must not be null"); this.sources = sources; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringIterableConfigurationPropertySource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringIterableConfigurationPropertySource.java index a2af54f1fe04..8e484ef3cad3 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringIterableConfigurationPropertySource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SpringIterableConfigurationPropertySource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -19,6 +19,7 @@ import java.util.Arrays; import java.util.Collections; import java.util.ConcurrentModificationException; +import java.util.HashMap; import java.util.HashSet; import java.util.Iterator; import java.util.LinkedHashMap; @@ -27,7 +28,6 @@ import java.util.Objects; import java.util.Set; import java.util.function.BiPredicate; -import java.util.function.Supplier; import java.util.stream.Stream; import org.springframework.boot.origin.Origin; @@ -38,6 +38,7 @@ import org.springframework.core.env.PropertySource; import org.springframework.core.env.StandardEnvironment; import org.springframework.core.env.SystemEnvironmentPropertySource; +import org.springframework.util.ConcurrentReferenceHashMap; /** * {@link ConfigurationPropertySource} backed by an {@link EnumerablePropertySource}. @@ -55,15 +56,20 @@ class SpringIterableConfigurationPropertySource extends SpringConfigurationPrope private final BiPredicate ancestorOfCheck; - private final SoftReferenceConfigurationPropertyCache cache; + private final SoftReferenceConfigurationPropertyCache cache; private volatile ConfigurationPropertyName[] configurationPropertyNames; - SpringIterableConfigurationPropertySource(EnumerablePropertySource propertySource, PropertyMapper... mappers) { - super(propertySource, mappers); + private final Map containsDescendantOfCache; + + SpringIterableConfigurationPropertySource(EnumerablePropertySource propertySource, + boolean systemEnvironmentSource, PropertyMapper... mappers) { + super(propertySource, systemEnvironmentSource, mappers); assertEnumerablePropertySource(); + boolean immutable = isImmutablePropertySource(); this.ancestorOfCheck = getAncestorOfCheck(mappers); - this.cache = new SoftReferenceConfigurationPropertyCache<>(isImmutablePropertySource()); + this.cache = new SoftReferenceConfigurationPropertyCache<>(immutable); + this.containsDescendantOfCache = (!systemEnvironmentSource) ? null : new ConcurrentReferenceHashMap<>(); } private BiPredicate getAncestorOfCheck( @@ -101,8 +107,8 @@ public ConfigurationProperty getConfigurationProperty(ConfigurationPropertyName if (configurationProperty != null) { return configurationProperty; } - for (String candidate : getMappings().getMapped(name)) { - Object value = getPropertySource().getProperty(candidate); + for (String candidate : getCache().getMapped(name)) { + Object value = getPropertySourceProperty(candidate); if (value != null) { Origin origin = PropertySourceOrigin.get(getPropertySource(), candidate); return ConfigurationProperty.of(this, name, value, origin); @@ -111,6 +117,11 @@ public ConfigurationProperty getConfigurationProperty(ConfigurationPropertyName return null; } + @Override + protected Object getSystemEnvironmentProperty(Map systemEnvironment, String name) { + return getCache().getSystemEnvironmentProperty(name); + } + @Override public Stream stream() { ConfigurationPropertyName[] names = getConfigurationPropertyNames(); @@ -129,42 +140,61 @@ public ConfigurationPropertyState containsDescendantOf(ConfigurationPropertyName return result; } if (this.ancestorOfCheck == PropertyMapper.DEFAULT_ANCESTOR_OF_CHECK) { - return getMappings().containsDescendantOf(name, this.ancestorOfCheck); + Set descendants = getCache().getDescendants(); + if (descendants != null) { + if (name.isEmpty() && !descendants.isEmpty()) { + return ConfigurationPropertyState.PRESENT; + } + return !descendants.contains(name) ? ConfigurationPropertyState.ABSENT + : ConfigurationPropertyState.PRESENT; + } } + result = (this.containsDescendantOfCache != null) ? this.containsDescendantOfCache.get(name) : null; + if (result == null) { + result = (!ancestorOfCheck(name)) ? ConfigurationPropertyState.ABSENT : ConfigurationPropertyState.PRESENT; + if (this.containsDescendantOfCache != null) { + this.containsDescendantOfCache.put(name, result); + } + } + return result; + } + + private boolean ancestorOfCheck(ConfigurationPropertyName name) { ConfigurationPropertyName[] candidates = getConfigurationPropertyNames(); for (ConfigurationPropertyName candidate : candidates) { if (candidate != null && this.ancestorOfCheck.test(name, candidate)) { - return ConfigurationPropertyState.PRESENT; + return true; } } - return ConfigurationPropertyState.ABSENT; + return false; } private ConfigurationPropertyName[] getConfigurationPropertyNames() { if (!isImmutablePropertySource()) { - return getMappings().getConfigurationPropertyNames(getPropertySource().getPropertyNames()); + return getCache().getConfigurationPropertyNames(getPropertySource().getPropertyNames()); } ConfigurationPropertyName[] configurationPropertyNames = this.configurationPropertyNames; if (configurationPropertyNames == null) { - configurationPropertyNames = getMappings() + configurationPropertyNames = getCache() .getConfigurationPropertyNames(getPropertySource().getPropertyNames()); this.configurationPropertyNames = configurationPropertyNames; } return configurationPropertyNames; } - private Mappings getMappings() { - return this.cache.get(this::createMappings, this::updateMappings); + private Cache getCache() { + return this.cache.get(this::createCache, this::updateCache); } - private Mappings createMappings() { - return new Mappings(getMappers(), isImmutablePropertySource(), - this.ancestorOfCheck == PropertyMapper.DEFAULT_ANCESTOR_OF_CHECK); + private Cache createCache() { + boolean immutable = isImmutablePropertySource(); + boolean captureDescendants = this.ancestorOfCheck == PropertyMapper.DEFAULT_ANCESTOR_OF_CHECK; + return new Cache(getMappers(), immutable, captureDescendants, isSystemEnvironmentSource()); } - private Mappings updateMappings(Mappings mappings) { - mappings.updateMappings(getPropertySource()::getPropertyNames); - return mappings; + private Cache updateCache(Cache cache) { + cache.update(getPropertySource()); + return cache; } private boolean isImmutablePropertySource() { @@ -183,7 +213,7 @@ protected EnumerablePropertySource getPropertySource() { return (EnumerablePropertySource) super.getPropertySource(); } - private static class Mappings { + private static class Cache { private static final ConfigurationPropertyName[] EMPTY_NAMES_ARRAY = {}; @@ -191,30 +221,26 @@ private static class Mappings { private final boolean immutable; - private final boolean trackDescendants; - - private volatile Map> mappings; - - private volatile Map reverseMappings; - - private volatile Map> descendants; + private final boolean captureDescendants; - private volatile ConfigurationPropertyName[] configurationPropertyNames; + private final boolean systemEnvironmentSource; - private volatile String[] lastUpdated; + private volatile Data data; - Mappings(PropertyMapper[] mappers, boolean immutable, boolean trackDescendants) { + Cache(PropertyMapper[] mappers, boolean immutable, boolean captureDescendants, + boolean systemEnvironmentSource) { this.mappers = mappers; this.immutable = immutable; - this.trackDescendants = trackDescendants; + this.captureDescendants = captureDescendants; + this.systemEnvironmentSource = systemEnvironmentSource; } - void updateMappings(Supplier propertyNames) { - if (this.mappings == null || !this.immutable) { + void update(EnumerablePropertySource propertySource) { + if (this.data == null || !this.immutable) { int count = 0; while (true) { try { - updateMappings(propertyNames.get()); + tryUpdate(propertySource); return; } catch (ConcurrentModificationException ex) { @@ -226,16 +252,21 @@ void updateMappings(Supplier propertyNames) { } } - private void updateMappings(String[] propertyNames) { - String[] lastUpdated = this.lastUpdated; + private void tryUpdate(EnumerablePropertySource propertySource) { + Data data = this.data; + String[] lastUpdated = (data != null) ? data.lastUpdated() : null; + String[] propertyNames = propertySource.getPropertyNames(); if (lastUpdated != null && Arrays.equals(lastUpdated, propertyNames)) { return; } int size = propertyNames.length; - Map> mappings = cloneOrCreate(this.mappings, size); - Map reverseMappings = cloneOrCreate(this.reverseMappings, size); - Map> descendants = cloneOrCreate(this.descendants, - size); + Map> mappings = cloneOrCreate( + (data != null) ? data.mappings() : null, size); + Map reverseMappings = cloneOrCreate( + (data != null) ? data.reverseMappings() : null, size); + Set descendants = (!this.captureDescendants) ? null : new HashSet<>(); + Map systemEnvironmentCopy = (!this.systemEnvironmentSource) ? null + : copySource(propertySource); for (PropertyMapper propertyMapper : this.mappers) { for (String propertyName : propertyNames) { if (!reverseMappings.containsKey(propertyName)) { @@ -243,30 +274,36 @@ private void updateMappings(String[] propertyNames) { if (configurationPropertyName != null && !configurationPropertyName.isEmpty()) { add(mappings, configurationPropertyName, propertyName); reverseMappings.put(propertyName, configurationPropertyName); - if (this.trackDescendants) { - addParents(descendants, configurationPropertyName); - } + addParents(descendants, configurationPropertyName); } } } } - this.mappings = mappings; - this.reverseMappings = reverseMappings; - this.descendants = descendants; - this.lastUpdated = this.immutable ? null : propertyNames; - this.configurationPropertyNames = this.immutable + ConfigurationPropertyName[] configurationPropertyNames = this.immutable ? reverseMappings.values().toArray(new ConfigurationPropertyName[0]) : null; + lastUpdated = this.immutable ? null : propertyNames; + this.data = new Data(mappings, reverseMappings, descendants, configurationPropertyNames, + systemEnvironmentCopy, lastUpdated); + } + + @SuppressWarnings("unchecked") + private HashMap copySource(EnumerablePropertySource propertySource) { + return new HashMap<>((Map) propertySource.getSource()); } private Map cloneOrCreate(Map source, int size) { return (source != null) ? new LinkedHashMap<>(source) : new LinkedHashMap<>(size); } - private void addParents(Map> descendants, - ConfigurationPropertyName name) { - ConfigurationPropertyName parent = name; + private void addParents(Set descendants, ConfigurationPropertyName name) { + if (descendants == null || name.isEmpty()) { + return; + } + ConfigurationPropertyName parent = name.getParent(); while (!parent.isEmpty()) { - add(descendants, parent, name); + if (!descendants.add(parent)) { + return; + } parent = parent.getParent(); } } @@ -276,15 +313,16 @@ private void add(Map> map, K key, T value) { } Set getMapped(ConfigurationPropertyName configurationPropertyName) { - return this.mappings.getOrDefault(configurationPropertyName, Collections.emptySet()); + return this.data.mappings().getOrDefault(configurationPropertyName, Collections.emptySet()); } ConfigurationPropertyName[] getConfigurationPropertyNames(String[] propertyNames) { - ConfigurationPropertyName[] names = this.configurationPropertyNames; + Data data = this.data; + ConfigurationPropertyName[] names = data.configurationPropertyNames(); if (names != null) { return names; } - Map reverseMappings = this.reverseMappings; + Map reverseMappings = data.reverseMappings(); if (reverseMappings == null || reverseMappings.isEmpty()) { return EMPTY_NAMES_ARRAY; } @@ -295,18 +333,19 @@ ConfigurationPropertyName[] getConfigurationPropertyNames(String[] propertyNames return names; } - ConfigurationPropertyState containsDescendantOf(ConfigurationPropertyName name, - BiPredicate ancestorOfCheck) { - if (name.isEmpty() && !this.descendants.isEmpty()) { - return ConfigurationPropertyState.PRESENT; - } - Set candidates = this.descendants.getOrDefault(name, Collections.emptySet()); - for (ConfigurationPropertyName candidate : candidates) { - if (ancestorOfCheck.test(name, candidate)) { - return ConfigurationPropertyState.PRESENT; - } - } - return ConfigurationPropertyState.ABSENT; + Set getDescendants() { + return this.data.descendants(); + } + + Object getSystemEnvironmentProperty(String name) { + return this.data.systemEnvironmentCopy().get(name); + } + + private record Data(Map> mappings, + Map reverseMappings, Set descendants, + ConfigurationPropertyName[] configurationPropertyNames, Map systemEnvironmentCopy, + String[] lastUpdated) { + } } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SystemEnvironmentPropertyMapper.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SystemEnvironmentPropertyMapper.java index aec68b63888d..2a74e2130743 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SystemEnvironmentPropertyMapper.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/context/properties/source/SystemEnvironmentPropertyMapper.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,7 +22,7 @@ import java.util.Locale; import java.util.function.BiPredicate; -import org.springframework.boot.context.properties.source.ConfigurationPropertyName.Form; +import org.springframework.boot.context.properties.source.ConfigurationPropertyName.ToStringFormat; /** * {@link PropertyMapper} for system environment variables. Names are mapped by removing @@ -42,44 +42,14 @@ final class SystemEnvironmentPropertyMapper implements PropertyMapper { @Override public List map(ConfigurationPropertyName configurationPropertyName) { - String name = convertName(configurationPropertyName); - String legacyName = convertLegacyName(configurationPropertyName); + String name = configurationPropertyName.toString(ToStringFormat.SYSTEM_ENVIRONMENT); + String legacyName = configurationPropertyName.toString(ToStringFormat.LEGACY_SYSTEM_ENVIRONMENT); if (name.equals(legacyName)) { return Collections.singletonList(name); } return Arrays.asList(name, legacyName); } - private String convertName(ConfigurationPropertyName name) { - return convertName(name, name.getNumberOfElements()); - } - - private String convertName(ConfigurationPropertyName name, int numberOfElements) { - StringBuilder result = new StringBuilder(); - for (int i = 0; i < numberOfElements; i++) { - if (!result.isEmpty()) { - result.append('_'); - } - result.append(name.getElement(i, Form.UNIFORM).toUpperCase(Locale.ENGLISH)); - } - return result.toString(); - } - - private String convertLegacyName(ConfigurationPropertyName name) { - StringBuilder result = new StringBuilder(); - for (int i = 0; i < name.getNumberOfElements(); i++) { - if (!result.isEmpty()) { - result.append('_'); - } - result.append(convertLegacyNameElement(name.getElement(i, Form.ORIGINAL))); - } - return result.toString(); - } - - private Object convertLegacyNameElement(String element) { - return element.replace('-', '_').toUpperCase(Locale.ENGLISH); - } - @Override public ConfigurationPropertyName map(String propertySourceName) { return convertName(propertySourceName); @@ -113,31 +83,11 @@ private boolean isAncestorOf(ConfigurationPropertyName name, ConfigurationProper } private boolean isLegacyAncestorOf(ConfigurationPropertyName name, ConfigurationPropertyName candidate) { - if (!hasDashedEntries(name)) { + if (!name.hasDashedElement()) { return false; } - ConfigurationPropertyName legacyCompatibleName = buildLegacyCompatibleName(name); + ConfigurationPropertyName legacyCompatibleName = name.asSystemEnvironmentLegacyName(); return legacyCompatibleName != null && legacyCompatibleName.isAncestorOf(candidate); } - private ConfigurationPropertyName buildLegacyCompatibleName(ConfigurationPropertyName name) { - StringBuilder legacyCompatibleName = new StringBuilder(); - for (int i = 0; i < name.getNumberOfElements(); i++) { - if (i != 0) { - legacyCompatibleName.append('.'); - } - legacyCompatibleName.append(name.getElement(i, Form.DASHED).replace('-', '.')); - } - return ConfigurationPropertyName.ofIfValid(legacyCompatibleName); - } - - boolean hasDashedEntries(ConfigurationPropertyName name) { - for (int i = 0; i < name.getNumberOfElements(); i++) { - if (name.getElement(i, Form.DASHED).indexOf('-') != -1) { - return true; - } - } - return false; - } - } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/ApplicationConversionService.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/ApplicationConversionService.java index a5ff4ce680b3..e76b6a3be0d4 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/ApplicationConversionService.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/ApplicationConversionService.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -17,12 +17,25 @@ package org.springframework.boot.convert; import java.lang.annotation.Annotation; -import java.util.LinkedHashSet; +import java.util.LinkedHashMap; +import java.util.Map; +import java.util.Optional; import java.util.Set; +import java.util.function.BiFunction; +import java.util.function.Consumer; +import java.util.function.Supplier; import org.springframework.beans.factory.ListableBeanFactory; +import org.springframework.beans.factory.annotation.BeanFactoryAnnotationUtils; +import org.springframework.beans.factory.config.BeanDefinition; +import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; +import org.springframework.context.ConfigurableApplicationContext; +import org.springframework.context.i18n.LocaleContextHolder; +import org.springframework.core.ResolvableType; import org.springframework.core.convert.ConversionService; import org.springframework.core.convert.TypeDescriptor; +import org.springframework.core.convert.converter.ConditionalConverter; +import org.springframework.core.convert.converter.ConditionalGenericConverter; import org.springframework.core.convert.converter.Converter; import org.springframework.core.convert.converter.ConverterFactory; import org.springframework.core.convert.converter.ConverterRegistry; @@ -37,6 +50,8 @@ import org.springframework.format.Printer; import org.springframework.format.support.DefaultFormattingConversionService; import org.springframework.format.support.FormattingConversionService; +import org.springframework.util.Assert; +import org.springframework.util.StringUtils; import org.springframework.util.StringValueResolver; /** @@ -49,10 +64,13 @@ * against registry instance. * * @author Phillip Webb + * @author Shixiong Guo * @since 2.0.0 */ public class ApplicationConversionService extends FormattingConversionService { + private static final ResolvableType STRING = ResolvableType.forClass(String.class); + private static volatile ApplicationConversionService sharedInstance; private final boolean unmodifiable; @@ -265,35 +283,302 @@ public static void addApplicationFormatters(FormatterRegistry registry) { } /** - * Add {@link GenericConverter}, {@link Converter}, {@link Printer}, {@link Parser} - * and {@link Formatter} beans from the specified context. + * Add {@link Printer}, {@link Parser}, {@link Formatter}, {@link Converter}, + * {@link ConverterFactory}, {@link GenericConverter}, and beans from the specified + * bean factory. * @param registry the service to register beans with * @param beanFactory the bean factory to get the beans from * @since 2.2.0 */ public static void addBeans(FormatterRegistry registry, ListableBeanFactory beanFactory) { - Set beans = new LinkedHashSet<>(); - beans.addAll(beanFactory.getBeansOfType(GenericConverter.class).values()); - beans.addAll(beanFactory.getBeansOfType(Converter.class).values()); - beans.addAll(beanFactory.getBeansOfType(Printer.class).values()); - beans.addAll(beanFactory.getBeansOfType(Parser.class).values()); - for (Object bean : beans) { - if (bean instanceof GenericConverter genericConverter) { - registry.addConverter(genericConverter); - } - else if (bean instanceof Converter converter) { - registry.addConverter(converter); - } - else if (bean instanceof Formatter formatter) { - registry.addFormatter(formatter); + addBeans(registry, beanFactory, null); + } + + /** + * Add {@link Printer}, {@link Parser}, {@link Formatter}, {@link Converter}, + * {@link ConverterFactory}, {@link GenericConverter}, and beans from the specified + * bean factory. + * @param registry the service to register beans with + * @param beanFactory the bean factory to get the beans from + * @param qualifier the qualifier required on the beans or {@code null} + * @return the beans that were added + * @since 3.5.0 + */ + public static Map addBeans(FormatterRegistry registry, ListableBeanFactory beanFactory, + String qualifier) { + ConfigurableListableBeanFactory configurableBeanFactory = getConfigurableListableBeanFactory(beanFactory); + Map beans = getBeans(beanFactory, qualifier); + beans.forEach((beanName, bean) -> { + BeanDefinition beanDefinition = (configurableBeanFactory != null) + ? configurableBeanFactory.getMergedBeanDefinition(beanName) : null; + ResolvableType type = (beanDefinition != null) ? beanDefinition.getResolvableType() : null; + addBean(registry, bean, type); + }); + return beans; + } + + private static ConfigurableListableBeanFactory getConfigurableListableBeanFactory(ListableBeanFactory beanFactory) { + if (beanFactory instanceof ConfigurableApplicationContext applicationContext) { + return applicationContext.getBeanFactory(); + } + if (beanFactory instanceof ConfigurableListableBeanFactory configurableListableBeanFactory) { + return configurableListableBeanFactory; + } + return null; + } + + private static Map getBeans(ListableBeanFactory beanFactory, String qualifier) { + Map beans = new LinkedHashMap<>(); + beans.putAll(getBeans(beanFactory, Printer.class, qualifier)); + beans.putAll(getBeans(beanFactory, Parser.class, qualifier)); + beans.putAll(getBeans(beanFactory, Formatter.class, qualifier)); + beans.putAll(getBeans(beanFactory, Converter.class, qualifier)); + beans.putAll(getBeans(beanFactory, ConverterFactory.class, qualifier)); + beans.putAll(getBeans(beanFactory, GenericConverter.class, qualifier)); + return beans; + } + + private static Map getBeans(ListableBeanFactory beanFactory, Class type, String qualifier) { + return (!StringUtils.hasLength(qualifier)) ? beanFactory.getBeansOfType(type) + : BeanFactoryAnnotationUtils.qualifiedBeansOfType(beanFactory, type, qualifier); + } + + static void addBean(FormatterRegistry registry, Object bean, ResolvableType beanType) { + if (bean instanceof GenericConverter converterBean) { + addBean(registry, converterBean, beanType, GenericConverter.class, registry::addConverter, (Runnable) null); + } + else if (bean instanceof Converter converterBean) { + addBean(registry, converterBean, beanType, Converter.class, registry::addConverter, + ConverterBeanAdapter::new); + } + else if (bean instanceof ConverterFactory converterBean) { + addBean(registry, converterBean, beanType, ConverterFactory.class, registry::addConverterFactory, + ConverterFactoryBeanAdapter::new); + } + else if (bean instanceof Formatter formatterBean) { + addBean(registry, formatterBean, beanType, Formatter.class, registry::addFormatter, () -> { + registry.addConverter(new PrinterBeanAdapter(formatterBean, beanType)); + registry.addConverter(new ParserBeanAdapter(formatterBean, beanType)); + }); + } + else if (bean instanceof Printer printerBean) { + addBean(registry, printerBean, beanType, Printer.class, registry::addPrinter, PrinterBeanAdapter::new); + } + else if (bean instanceof Parser parserBean) { + addBean(registry, parserBean, beanType, Parser.class, registry::addParser, ParserBeanAdapter::new); + } + } + + private static void addBean(FormatterRegistry registry, B bean, ResolvableType beanType, Class type, + Consumer standardRegistrar, BiFunction> beanAdapterFactory) { + addBean(registry, bean, beanType, type, standardRegistrar, + () -> registry.addConverter(beanAdapterFactory.apply(bean, beanType))); + } + + private static void addBean(FormatterRegistry registry, B bean, ResolvableType beanType, Class type, + Consumer standardRegistrar, Runnable beanAdapterRegistrar) { + if (beanType != null && beanAdapterRegistrar != null + && ResolvableType.forInstance(bean).as(type).hasUnresolvableGenerics()) { + beanAdapterRegistrar.run(); + return; + } + standardRegistrar.accept(bean); + } + + /** + * Base class for adapters that adapt a bean to a {@link GenericConverter}. + * + * @param the base type of the bean + */ + abstract static class BeanAdapter implements ConditionalGenericConverter { + + private final B bean; + + private final ResolvableTypePair types; + + BeanAdapter(B bean, ResolvableType beanType) { + Assert.isInstanceOf(beanType.toClass(), bean); + ResolvableType type = ResolvableType.forClass(getClass()).as(BeanAdapter.class).getGeneric(); + ResolvableType[] generics = beanType.as(type.toClass()).getGenerics(); + this.bean = bean; + this.types = getResolvableTypePair(generics); + } + + protected ResolvableTypePair getResolvableTypePair(ResolvableType[] generics) { + return new ResolvableTypePair(generics[0], generics[1]); + } + + protected B bean() { + return this.bean; + } + + @Override + public Set getConvertibleTypes() { + return Set.of(new ConvertiblePair(this.types.source().toClass(), this.types.target().toClass())); + } + + @Override + public boolean matches(TypeDescriptor sourceType, TypeDescriptor targetType) { + return (this.types.target().toClass() == targetType.getObjectType() + && matchesTargetType(targetType.getResolvableType())); + } + + private boolean matchesTargetType(ResolvableType targetType) { + ResolvableType ours = this.types.target(); + return targetType.getType() instanceof Class || targetType.isAssignableFrom(ours) + || this.types.target().hasUnresolvableGenerics(); + } + + protected final boolean conditionalConverterCandidateMatches(Object conditionalConverterCandidate, + TypeDescriptor sourceType, TypeDescriptor targetType) { + return (conditionalConverterCandidate instanceof ConditionalConverter conditionalConverter) + ? conditionalConverter.matches(sourceType, targetType) : true; + } + + @SuppressWarnings({ "unchecked", "rawtypes" }) + protected final Object convert(Object source, TypeDescriptor targetType, Converter converter) { + return (source != null) ? ((Converter) converter).convert(source) : convertNull(targetType); + } + + private Object convertNull(TypeDescriptor targetType) { + return (targetType.getObjectType() != Optional.class) ? null : Optional.empty(); + } + + @Override + public String toString() { + return this.types + " : " + this.bean; + } + + } + + /** + * Adapts a {@link Printer} bean to a {@link GenericConverter}. + */ + static class PrinterBeanAdapter extends BeanAdapter> { + + PrinterBeanAdapter(Printer bean, ResolvableType beanType) { + super(bean, beanType); + } + + @Override + protected ResolvableTypePair getResolvableTypePair(ResolvableType[] generics) { + return new ResolvableTypePair(generics[0], STRING); + } + + @Override + public Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType) { + return (source != null) ? print(source) : ""; + } + + @SuppressWarnings("unchecked") + private String print(Object object) { + return ((Printer) bean()).print(object, LocaleContextHolder.getLocale()); + } + + } + + /** + * Adapts a {@link Parser} bean to a {@link GenericConverter}. + */ + static class ParserBeanAdapter extends BeanAdapter> { + + ParserBeanAdapter(Parser bean, ResolvableType beanType) { + super(bean, beanType); + } + + @Override + protected ResolvableTypePair getResolvableTypePair(ResolvableType[] generics) { + return new ResolvableTypePair(STRING, generics[0]); + } + + @Override + public Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType) { + String text = (String) source; + return (!StringUtils.hasText(text)) ? null : parse(text); + } + + private Object parse(String text) { + try { + return bean().parse(text, LocaleContextHolder.getLocale()); } - else if (bean instanceof Printer printer) { - registry.addPrinter(printer); + catch (IllegalArgumentException ex) { + throw ex; } - else if (bean instanceof Parser parser) { - registry.addParser(parser); + catch (Throwable ex) { + throw new IllegalArgumentException("Parse attempt failed for value [" + text + "]", ex); } } + + } + + /** + * Adapts a {@link Converter} bean to a {@link GenericConverter}. + */ + static final class ConverterBeanAdapter extends BeanAdapter> { + + ConverterBeanAdapter(Converter bean, ResolvableType beanType) { + super(bean, beanType); + } + + @Override + public boolean matches(TypeDescriptor sourceType, TypeDescriptor targetType) { + return super.matches(sourceType, targetType) + && conditionalConverterCandidateMatches(bean(), sourceType, targetType); + } + + @Override + public Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType) { + return convert(source, targetType, bean()); + } + + } + + /** + * Adapts a {@link ConverterFactory} bean to a {@link GenericConverter}. + */ + private static final class ConverterFactoryBeanAdapter extends BeanAdapter> { + + ConverterFactoryBeanAdapter(ConverterFactory bean, ResolvableType beanType) { + super(bean, beanType); + } + + @Override + public boolean matches(TypeDescriptor sourceType, TypeDescriptor targetType) { + return super.matches(sourceType, targetType) + && conditionalConverterCandidateMatches(bean(), sourceType, targetType) + && conditionalConverterCandidateMatches(getConverter(targetType::getType), sourceType, targetType); + } + + @Override + public Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType) { + return convert(source, targetType, getConverter(targetType::getObjectType)); + } + + @SuppressWarnings({ "unchecked", "rawtypes" }) + private Converter getConverter(Supplier> typeSupplier) { + return ((ConverterFactory) bean()).getConverter(typeSupplier.get()); + } + + } + + /** + * Convertible type information as extracted from bean generics. + * + * @param source the source type + * @param target the target type + */ + record ResolvableTypePair(ResolvableType source, ResolvableType target) { + + ResolvableTypePair { + Assert.notNull(source.resolve(), "'source' cannot be resolved"); + Assert.notNull(target.resolve(), "'target' cannot be resolved"); + } + + @Override + public final String toString() { + return source() + " -> " + target(); + } + } } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DelimitedStringToArrayConverter.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DelimitedStringToArrayConverter.java index e2b4237924ba..4ece3986fe8d 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DelimitedStringToArrayConverter.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DelimitedStringToArrayConverter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -36,7 +36,7 @@ final class DelimitedStringToArrayConverter implements ConditionalGenericConvert private final ConversionService conversionService; DelimitedStringToArrayConverter(ConversionService conversionService) { - Assert.notNull(conversionService, "ConversionService must not be null"); + Assert.notNull(conversionService, "'conversionService' must not be null"); this.conversionService = conversionService; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DelimitedStringToCollectionConverter.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DelimitedStringToCollectionConverter.java index 7f5281a28f9a..d9b1f2e29b37 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DelimitedStringToCollectionConverter.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DelimitedStringToCollectionConverter.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2019 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -39,7 +39,7 @@ final class DelimitedStringToCollectionConverter implements ConditionalGenericCo private final ConversionService conversionService; DelimitedStringToCollectionConverter(ConversionService conversionService) { - Assert.notNull(conversionService, "ConversionService must not be null"); + Assert.notNull(conversionService, "'conversionService' must not be null"); this.conversionService = conversionService; } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DurationStyle.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DurationStyle.java index 61833f0c97c8..46aabde4e88b 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DurationStyle.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/DurationStyle.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2023 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -162,7 +162,7 @@ public static Duration detectAndParse(String value, ChronoUnit unit) { * @throws IllegalArgumentException if the value is not a known style */ public static DurationStyle detect(String value) { - Assert.notNull(value, "Value must not be null"); + Assert.notNull(value, "'value' must not be null"); for (DurationStyle candidate : values()) { if (candidate.matches(value)) { return candidate; diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/PeriodStyle.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/PeriodStyle.java index ea43e8b7b04f..820d7e360b18 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/PeriodStyle.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/convert/PeriodStyle.java @@ -1,5 +1,5 @@ /* - * Copyright 2002-2022 the original author or authors. + * Copyright 2002-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -204,7 +204,7 @@ public static Period detectAndParse(String value, ChronoUnit unit) { * @throws IllegalArgumentException if the value is not a known style */ public static PeriodStyle detect(String value) { - Assert.notNull(value, "Value must not be null"); + Assert.notNull(value, "'value' must not be null"); for (PeriodStyle candidate : values()) { if (candidate.matches(value)) { return candidate; @@ -264,7 +264,7 @@ private boolean isZero(Period value) { } private int intValue(Period value) { - Assert.notNull(this.intValue, () -> "intValue cannot be extracted from " + name()); + Assert.state(this.intValue != null, () -> "intValue cannot be extracted from " + name()); return this.intValue.apply(value); } diff --git a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/env/ConfigTreePropertySource.java b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/env/ConfigTreePropertySource.java index becea918c401..37526035693e 100644 --- a/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/env/ConfigTreePropertySource.java +++ b/spring-boot-project/spring-boot/src/main/java/org/springframework/boot/env/ConfigTreePropertySource.java @@ -1,5 +1,5 @@ /* - * Copyright 2012-2024 the original author or authors. + * Copyright 2012-2025 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -43,8 +43,8 @@ import org.springframework.core.env.EnumerablePropertySource; import org.springframework.core.env.Environment; import org.springframework.core.env.PropertySource; +import org.springframework.core.io.FileSystemResource; import org.springframework.core.io.InputStreamSource; -import org.springframework.core.io.PathResource; import org.springframework.core.io.Resource; import org.springframework.util.Assert; import org.springframework.util.FileCopyUtils; @@ -105,8 +105,10 @@ public ConfigTreePropertySource(String name, Path sourceDirectory, Option... opt private ConfigTreePropertySource(String name, Path sourceDirectory, Set