Spring's @ActiveProfiles can be used inside your own composed annotations (meta-annotations). This lets you define reusable testing presets (e.g., a specific configuration and set of profiles) and apply them to multiple test classes without duplication.
Use Cases:
- Standardize environment presets like dev/mysql vs prod/postgres across many tests.
- Reduce boilerplate by colocating @SpringJUnitConfig and @ActiveProfiles in a single custom annotation.
- Make intent explicit with domain-specific annotations (e.g., @DevMySqlTest).
Example
Configuration
package com.logicbig.example;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Profile;
import org.springframework.jdbc.datasource.DriverManagerDataSource;
import javax.sql.DataSource;
@Configuration
public class AppConfig {
// Minimal DataSource beans for demo purposes only;
// no real DB connectivity required.
@Bean
@Profile({"dev", "mysql"})
public DataSource devMySqlDataSource() {
DriverManagerDataSource ds = new DriverManagerDataSource();
ds.setUrl("jdbc:mysql://localhost:3306/demo");
ds.setUsername("demo");
ds.setPassword("demo");
return ds;
}
@Bean
@Profile({"prod", "postgres"})
public DataSource prodPostgresDataSource() {
DriverManagerDataSource ds = new DriverManagerDataSource();
ds.setUrl("jdbc:postgresql://localhost:5432/demo");
ds.setUsername("demo");
ds.setPassword("demo");
return ds;
}
}
Custom composed annotations
package com.logicbig.example;
import org.springframework.test.context.ActiveProfiles;
import org.springframework.test.context.junit.jupiter.SpringJUnitConfig;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@SpringJUnitConfig(classes = AppConfig.class)
@ActiveProfiles({"dev", "mysql"})
public @interface DevMySqlTest {
}
package com.logicbig.example;
import org.springframework.test.context.ActiveProfiles;
import org.springframework.test.context.junit.jupiter.SpringJUnitConfig;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@SpringJUnitConfig(classes = AppConfig.class)
@ActiveProfiles({"prod", "postgres"})
public @interface ProdPostgresTest {
}
JUnit tests using the composed annotations
package com.logicbig.example;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.env.Environment;
import org.springframework.core.env.Profiles;
import java.util.Arrays;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertTrue;
@DevMySqlTest
public class UserServiceTest {
@Autowired
private Environment environment;
@Test
public void testProfiles() {
String[] profiles = environment.getActiveProfiles();
System.out.println("Active profiles (UserServiceTest): " +
Arrays.toString(profiles));
// Assertions
assertTrue(Arrays.asList(profiles).contains("dev"));
assertTrue(Arrays.asList(profiles).contains("mysql"));
assertFalse(Arrays.asList(profiles).contains("prod"));
assertFalse(Arrays.asList(profiles).contains("postgres"));
// Environment convenience with Profiles.of
assertTrue(environment.acceptsProfiles(
Profiles.of("dev & mysql")));
assertFalse(environment.acceptsProfiles(Profiles.of("prod")));
}
}
OutputD:\example-projects\spring-core-testing\spring-active-profiles-meta-annotation-example>mvn test -Dtest=UserServiceTest [INFO] Scanning for projects... [INFO] [INFO] --< com.logicbig.example:spring-active-profiles-meta-annotation-example >-- [INFO] Building spring-active-profiles-meta-annotation-example 1.0-SNAPSHOT [INFO] from pom.xml [INFO] --------------------------------[ jar ]--------------------------------- [INFO] [INFO] --- resources:3.3.1:resources (default-resources) @ spring-active-profiles-meta-annotation-example --- [WARNING] Using platform encoding (UTF-8 actually) to copy filtered resources, i.e. build is platform dependent! [INFO] skip non existing resourceDirectory D:\example-projects\spring-core-testing\spring-active-profiles-meta-annotation-example\src\main\resources [INFO] [INFO] --- compiler:3.14.1:compile (default-compile) @ spring-active-profiles-meta-annotation-example --- [INFO] No sources to compile [INFO] [INFO] --- resources:3.3.1:testResources (default-testResources) @ spring-active-profiles-meta-annotation-example --- [WARNING] Using platform encoding (UTF-8 actually) to copy filtered resources, i.e. build is platform dependent! [INFO] skip non existing resourceDirectory D:\example-projects\spring-core-testing\spring-active-profiles-meta-annotation-example\src\test\resources [INFO] [INFO] --- compiler:3.14.1:testCompile (default-testCompile) @ spring-active-profiles-meta-annotation-example --- [INFO] Recompiling the module because of changed source code. [INFO] Compiling 6 source files with javac [debug target 25] to target\test-classes [INFO] [INFO] --- surefire:3.5.0:test (default-test) @ spring-active-profiles-meta-annotation-example --- [INFO] Using auto detected provider org.apache.maven.surefire.junitplatform.JUnitPlatformProvider [WARNING] file.encoding cannot be set as system property, use <argLine>-Dfile.encoding=...</argLine> instead [INFO] [INFO] ------------------------------------------------------- [INFO] T E S T S [INFO] ------------------------------------------------------- Active profiles (UserServiceTest): [dev, mysql] [INFO] +--com.logicbig.example.UserServiceTest - 0.424 ss [INFO] | '-- [OK] testProfiles - 0.037 ss [INFO] [INFO] Results: [INFO] [INFO] Tests run: 1, Failures: 0, Errors: 0, Skipped: 0 [INFO] [INFO] ------------------------------------------------------------------------ [INFO] BUILD SUCCESS [INFO] ------------------------------------------------------------------------ [INFO] Total time: 2.793 s [INFO] Finished at: 2026-01-31T07:24:23+08:00 [INFO] ------------------------------------------------------------------------
package com.logicbig.example;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.env.Environment;
import org.springframework.core.env.Profiles;
import javax.sql.DataSource;
import java.util.Arrays;
import static org.junit.jupiter.api.Assertions.*;
@DevMySqlTest
public class OrderServiceTest {
@Autowired
private DataSource dataSource;
@Autowired
private Environment environment;
@Test
public void testOrder() {
// has [dev, mysql] profiles
String[] profiles = environment.getActiveProfiles();
System.out.println("Active profiles (OrderServiceTest): " +
Arrays.toString(profiles));
assertNotNull(dataSource, "DataSource should be autowired "
+ "for dev/mysql profile combination");
assertTrue(Arrays.asList(profiles).contains("dev"));
assertTrue(Arrays.asList(profiles).contains("mysql"));
assertFalse(Arrays.asList(profiles).contains("prod"));
assertFalse(Arrays.asList(profiles).contains("postgres"));
assertTrue(environment.acceptsProfiles(Profiles.of("dev & mysql")));
assertFalse(environment.acceptsProfiles(
Profiles.of("prod | postgres")));
}
}
OutputD:\example-projects\spring-core-testing\spring-active-profiles-meta-annotation-example>mvn test -Dtest=OrderServiceTest [INFO] Scanning for projects... [INFO] [INFO] --< com.logicbig.example:spring-active-profiles-meta-annotation-example >-- [INFO] Building spring-active-profiles-meta-annotation-example 1.0-SNAPSHOT [INFO] from pom.xml [INFO] --------------------------------[ jar ]--------------------------------- [INFO] [INFO] --- resources:3.3.1:resources (default-resources) @ spring-active-profiles-meta-annotation-example --- [WARNING] Using platform encoding (UTF-8 actually) to copy filtered resources, i.e. build is platform dependent! [INFO] skip non existing resourceDirectory D:\example-projects\spring-core-testing\spring-active-profiles-meta-annotation-example\src\main\resources [INFO] [INFO] --- compiler:3.14.1:compile (default-compile) @ spring-active-profiles-meta-annotation-example --- [INFO] No sources to compile [INFO] [INFO] --- resources:3.3.1:testResources (default-testResources) @ spring-active-profiles-meta-annotation-example --- [WARNING] Using platform encoding (UTF-8 actually) to copy filtered resources, i.e. build is platform dependent! [INFO] skip non existing resourceDirectory D:\example-projects\spring-core-testing\spring-active-profiles-meta-annotation-example\src\test\resources [INFO] [INFO] --- compiler:3.14.1:testCompile (default-testCompile) @ spring-active-profiles-meta-annotation-example --- [INFO] Nothing to compile - all classes are up to date. [INFO] [INFO] --- surefire:3.5.0:test (default-test) @ spring-active-profiles-meta-annotation-example --- [INFO] Using auto detected provider org.apache.maven.surefire.junitplatform.JUnitPlatformProvider [WARNING] file.encoding cannot be set as system property, use <argLine>-Dfile.encoding=...</argLine> instead [INFO] [INFO] ------------------------------------------------------- [INFO] T E S T S [INFO] ------------------------------------------------------- Active profiles (OrderServiceTest): [dev, mysql] [INFO] +--com.logicbig.example.OrderServiceTest - 0.489 ss [INFO] | '-- [OK] testOrder - 0.039 ss [INFO] [INFO] Results: [INFO] [INFO] Tests run: 1, Failures: 0, Errors: 0, Skipped: 0 [INFO] [INFO] ------------------------------------------------------------------------ [INFO] BUILD SUCCESS [INFO] ------------------------------------------------------------------------ [INFO] Total time: 2.658 s [INFO] Finished at: 2026-01-31T07:24:33+08:00 [INFO] ------------------------------------------------------------------------
package com.logicbig.example;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.env.Environment;
import org.springframework.core.env.Profiles;
import java.util.Arrays;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertTrue;
@ProdPostgresTest
public class ReportServiceTest {
@Autowired
private Environment environment;
@Test
public void testReport() {
// has [prod, postgres] profiles
String[] profiles = environment.getActiveProfiles();
System.out.println("Active profiles (ReportServiceTest): " +
Arrays.toString(profiles));
assertTrue(Arrays.asList(profiles).contains("prod"));
assertTrue(Arrays.asList(profiles).contains("postgres"));
assertFalse(Arrays.asList(profiles).contains("dev"));
assertFalse(Arrays.asList(profiles).contains("mysql"));
assertTrue(environment.acceptsProfiles(
Profiles.of("prod & postgres")));
assertFalse(environment.acceptsProfiles(
Profiles.of("dev | mysql")));
}
}
OutputD:\example-projects\spring-core-testing\spring-active-profiles-meta-annotation-example>mvn test -Dtest=ReportServiceTest [INFO] Scanning for projects... [INFO] [INFO] --< com.logicbig.example:spring-active-profiles-meta-annotation-example >-- [INFO] Building spring-active-profiles-meta-annotation-example 1.0-SNAPSHOT [INFO] from pom.xml [INFO] --------------------------------[ jar ]--------------------------------- [INFO] [INFO] --- resources:3.3.1:resources (default-resources) @ spring-active-profiles-meta-annotation-example --- [WARNING] Using platform encoding (UTF-8 actually) to copy filtered resources, i.e. build is platform dependent! [INFO] skip non existing resourceDirectory D:\example-projects\spring-core-testing\spring-active-profiles-meta-annotation-example\src\main\resources [INFO] [INFO] --- compiler:3.14.1:compile (default-compile) @ spring-active-profiles-meta-annotation-example --- [INFO] No sources to compile [INFO] [INFO] --- resources:3.3.1:testResources (default-testResources) @ spring-active-profiles-meta-annotation-example --- [WARNING] Using platform encoding (UTF-8 actually) to copy filtered resources, i.e. build is platform dependent! [INFO] skip non existing resourceDirectory D:\example-projects\spring-core-testing\spring-active-profiles-meta-annotation-example\src\test\resources [INFO] [INFO] --- compiler:3.14.1:testCompile (default-testCompile) @ spring-active-profiles-meta-annotation-example --- [INFO] Nothing to compile - all classes are up to date. [INFO] [INFO] --- surefire:3.5.0:test (default-test) @ spring-active-profiles-meta-annotation-example --- [INFO] Using auto detected provider org.apache.maven.surefire.junitplatform.JUnitPlatformProvider [WARNING] file.encoding cannot be set as system property, use <argLine>-Dfile.encoding=...</argLine> instead [INFO] [INFO] ------------------------------------------------------- [INFO] T E S T S [INFO] ------------------------------------------------------- Active profiles (ReportServiceTest): [prod, postgres] [INFO] +--com.logicbig.example.ReportServiceTest - 0.441 ss [INFO] | '-- [OK] testReport - 0.040 ss [INFO] [INFO] Results: [INFO] [INFO] Tests run: 1, Failures: 0, Errors: 0, Skipped: 0 [INFO] [INFO] ------------------------------------------------------------------------ [INFO] BUILD SUCCESS [INFO] ------------------------------------------------------------------------ [INFO] Total time: 3.283 s [INFO] Finished at: 2026-01-31T07:24:43+08:00 [INFO] ------------------------------------------------------------------------
Example ProjectDependencies and Technologies Used: - spring-context 7.0.3 (Spring Context)
Version Compatibility: 5.1.0.RELEASE - 7.0.3 Version compatibilities of spring-context with this example: Versions in green have been tested.
- spring-jdbc 7.0.3 (Spring JDBC)
- spring-test 7.0.3 (Spring TestContext Framework)
- junit-jupiter-engine 6.0.2 (Module "junit-jupiter-engine" of JUnit)
- JDK 25
- Maven 3.9.11
|