Query by Spec (QBS) is a user-friendly querying approach that allows you to dynamically build query conditions using specifications. With the QBS interface, you can execute query statements easily.
Getting Started
By adding the dependency in your pom.xml file, the specification-mapper-starter will automatically configure everything during the Spring Boot startup process, allowing you to start using it without any additional configuration. The starter includes the following features:
The auto-configuration is enabled by default, and you can control it through the spec.mapper.enabled property in your application’s properties file. To disable the auto-configuration, you can use the following configuration:
spec:mapper:enabled:false
1 - Query by Spec
Introduction and usage of Query by Spec (QBS)
Query by Spec (QBS) provides the QueryBySpecExecutor<T> interface, which includes several query methods:
publicinterfaceQueryBySpecExecutor<T>{List<T>findBySpec(Objectspec);List<T>findBySpec(Objectspec,Sortsort);Page<T>findBySpec(Objectspec,Pageablepageable);// … more functionality omitted.}
To use these methods, you simply need to extend QueryBySpecExecutor<T> in your existing repository interface:
By inheriting QueryBySpecExecutor<T>, you can directly use the query methods in your repository interface, making it easy to perform queries using specifications.
Customize Base Repository
During the configuration process, QBS automatically configures the Spring Data JPA Base Repository. The default implementation is DefaultQueryBySpecExecutor.
However, since Java only supports single inheritance, and to allow your application to retain its original parent Base Repository, QBS provides an extension point called QueryBySpecExecutorAdapter.
Depending on your application’s needs, you can choose to either extend DefaultQueryBySpecExecutor or implement QueryBySpecExecutorAdapter to customize the Base Repository. For example:
classMyRepositoryImpl<T,ID>extendsSimpleJpaRepository<T,ID>implementsQueryBySpecExecutorAdapter<T>{@Setter@GetterprivateSpecMapperspecMapper;privatefinalEntityManagerentityManager;MyRepositoryImpl(JpaEntityInformationentityInformation,EntityManagerentityManager){super(entityInformation,entityManager);// Keep the EntityManager around to be used from the newly introduced methods.this.entityManager=entityManager;}@OverridepublicClass<T>getDomainClass(){returnsuper.getDomainClass();}@Transactionalpublic<SextendsT>SmySave(Sentity){// implementation goes here}}
You can configure your custom Base Repository by setting the spec.mapper.repository-base-class property in your application’s properties file, specifying the full package name of your custom base repository, like this:
During application startup, the starter automatically configures a Default SpecMapper and registers it as a Spring @Bean, allowing you to retrieve it via Autowired.
@AutowiredSpecMapperspecMapper;
For example, if you want to enhance the specifications before performing the query, you can use the SpecMapper as follows:
classPersonService{@AutowiredSpecMapperspecMapper;@AutowiredPersonRepositorypersonRepository;List<Person>getPersonByCriteria(PersonCriteriacriteria){varspec=specMapper.toSpec(criteria);// Perform additional operations on the spec, ex:// spec = spec.and((root, query, criteriaBuilder) -> {// ...// });returnpersonRepository.findAll(spec);}}
In the above example, the SpecMapper is injected into the PersonService, allowing you to convert the criteria into a specification using specMapper.toSpec(). You can then modify the spec as needed before passing it to the personRepository for querying.
Configuration
The starter provides multiple ways to adjust the configuration of the Default SpecMapper.
SpecificationResolver
SpecificationResolver allows you to add custom Spec annotations. Simply register your custom implementation as a Spring @Bean, and it will be automatically detected and configured during application startup.
If your SpecificationResolver needs access to the SpecMapper itself, you can wrap it in a SpecificationResolverCodecBuilder. This way, the SpecCodec, which is the interface of SpecMapper, will be passed in when constructing the resolver. Here’s an example:
@ConfigurationclassMyConfig{@BeanSpecificationResolverCodecBuildermyResolver(){returnMySpecificationResolver::new;}}classMySpecificationResolverimplementsSpecificationResolver{privatefinalSpecCodeccodec;MySpecificationResolver(SpecCodeccodec){// Keep the SpecCodec around to be used.this.codec=codec;}// implementation goes here}
In the above example, the MySpecificationResolver is constructed with the SpecCodec provided by the SpecMapper. This allows you to access and utilize the SpecMapper functionality within your custom resolver.
SkippingStrategy
SkippingStrategy defines rules for skipping specific fields. By registering your custom implementation as a Spring @Bean, it will be automatically detected and added to the Default SpecMapper during application startup.
As described in Logging, different Logger Name strategies are available. You can enable impersonation mode using the spec.mapper.impersonate-logger property. By default, this setting is disabled. To enable it:
spec:mapper:# Whether to impersonate the logger of the actual object being processed, off by defaultimpersonate-logger:true
For full customization, register your own ASTWriterFactory implementation as a Spring @Bean, and it will be automatically detected and added to the Default SpecMapper.
You can also fully customize SpecMapper by registering your own implementation as a Spring @Bean, which will take precedence over the default configuration.