Skip to content

Commit 4e07b85

Browse files
mp911dechristophstrobl
authored andcommitted
Add support for providing default Querydsl bindings.
We now consider default Querydsl bindings when registering a DefaultQuerydslBinderCustomizer bean. The default bindings are applied before applying type-specific bindings. Closes #206. Original Pull Request: #2292
1 parent e3a8edf commit 4e07b85

File tree

4 files changed

+95
-2
lines changed

4 files changed

+95
-2
lines changed

src/main/asciidoc/repositories.adoc

+2
Original file line numberDiff line numberDiff line change
@@ -1667,6 +1667,8 @@ interface UserRepository extends CrudRepository<User, String>,
16671667
<5> Exclude the `password` property from `Predicate` resolution.
16681668
====
16691669

1670+
Additionally, you can register a `DefaultQuerydslBinderCustomizer` bean to apply default Querydsl bindings before applying specific bindings from the repository or `@QuerydslPredicate`.
1671+
16701672
[[core.repository-populators]]
16711673
=== Repository Populators
16721674

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
/*
2+
* Copyright 2021 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.querydsl.binding;
17+
18+
import com.querydsl.core.types.EntityPath;
19+
20+
/**
21+
* A component for {@link QuerydslBindings} customization acting as default customizer the given entity path regardless
22+
* of the domain type. Instances can be registered with the application context to be applied.
23+
*
24+
* @author Mark Paluch
25+
* @since 2.5
26+
*/
27+
public interface DefaultQuerydslBinderCustomizer extends QuerydslBinderCustomizer<EntityPath<?>> {}

src/main/java/org/springframework/data/querydsl/binding/QuerydslBindingsFactory.java

+32-2
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,10 @@
1515
*/
1616
package org.springframework.data.querydsl.binding;
1717

18+
import java.util.List;
1819
import java.util.Map;
1920
import java.util.Optional;
21+
import java.util.stream.Collectors;
2022

2123
import org.springframework.beans.BeanUtils;
2224
import org.springframework.beans.BeansException;
@@ -37,6 +39,7 @@
3739
*
3840
* @author Oliver Gierke
3941
* @author Christoph Strobl
42+
* @author Mark Paluch
4043
* @since 1.11
4144
*/
4245
public class QuerydslBindingsFactory implements ApplicationContextAware {
@@ -48,6 +51,7 @@ public class QuerydslBindingsFactory implements ApplicationContextAware {
4851

4952
private Optional<AutowireCapableBeanFactory> beanFactory;
5053
private Optional<Repositories> repositories;
54+
private QuerydslBinderCustomizer<EntityPath<?>> defaultCustomizer;
5155

5256
/**
5357
* Creates a new {@link QuerydslBindingsFactory} using the given {@link EntityPathResolver}.
@@ -62,6 +66,7 @@ public QuerydslBindingsFactory(EntityPathResolver entityPathResolver) {
6266
this.entityPaths = new ConcurrentReferenceHashMap<>();
6367
this.beanFactory = Optional.empty();
6468
this.repositories = Optional.empty();
69+
this.defaultCustomizer = NoOpCustomizer.INSTANCE;
6570
}
6671

6772
/*
@@ -73,6 +78,7 @@ public void setApplicationContext(ApplicationContext applicationContext) throws
7378

7479
this.beanFactory = Optional.of(applicationContext.getAutowireCapableBeanFactory());
7580
this.repositories = Optional.of(new Repositories(applicationContext));
81+
this.defaultCustomizer = findDefaultCustomizer();
7682
}
7783

7884
/**
@@ -126,6 +132,7 @@ private QuerydslBindings createBindingsFor(TypeInformation<?> domainType,
126132
EntityPath<?> path = verifyEntityPathPresent(domainType);
127133

128134
QuerydslBindings bindings = new QuerydslBindings();
135+
defaultCustomizer.customize(bindings, path);
129136
findCustomizerForDomainType(customizer, domainType.getType()).customize(bindings, path);
130137

131138
return bindings;
@@ -151,9 +158,32 @@ private EntityPath<?> verifyEntityPathPresent(TypeInformation<?> candidate) {
151158
});
152159
}
153160

161+
/**
162+
* Obtains registered {@link DefaultQuerydslBinderCustomizer} instances from the
163+
* {@link org.springframework.beans.factory.BeanFactory}.
164+
*
165+
* @return
166+
*/
167+
private QuerydslBinderCustomizer<EntityPath<?>> findDefaultCustomizer() {
168+
return beanFactory.map(this::getDefaultQuerydslBinderCustomizer).orElse(NoOpCustomizer.INSTANCE);
169+
}
170+
171+
private QuerydslBinderCustomizer<EntityPath<?>> getDefaultQuerydslBinderCustomizer(
172+
AutowireCapableBeanFactory beanFactory) {
173+
174+
List<DefaultQuerydslBinderCustomizer> customizers = beanFactory
175+
.getBeanProvider(DefaultQuerydslBinderCustomizer.class).stream().collect(Collectors.toList());
176+
177+
return (bindings, root) -> {
178+
for (DefaultQuerydslBinderCustomizer defaultQuerydslBinderCustomizer : customizers) {
179+
defaultQuerydslBinderCustomizer.customize(bindings, root);
180+
}
181+
};
182+
}
183+
154184
/**
155185
* Obtains the {@link QuerydslBinderCustomizer} for the given domain type. Will inspect the given annotation for a
156-
* dedicatedly configured one or consider the domain types's repository.
186+
* dedicated configured one or consider the domain type's repository.
157187
*
158188
* @param annotation
159189
* @param domainType
@@ -194,7 +224,7 @@ private QuerydslBinderCustomizer<EntityPath<?>> createQuerydslBinderCustomizer(
194224
}).orElseGet(() -> BeanUtils.instantiateClass(type));
195225
}
196226

197-
private static enum NoOpCustomizer implements QuerydslBinderCustomizer<EntityPath<?>> {
227+
private enum NoOpCustomizer implements QuerydslBinderCustomizer<EntityPath<?>> {
198228

199229
INSTANCE;
200230

src/test/java/org/springframework/data/querydsl/binding/QuerydslBindingsFactoryUnitTests.java

+34
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,9 @@
2323

2424
import org.junit.jupiter.api.BeforeEach;
2525
import org.junit.jupiter.api.Test;
26+
2627
import org.springframework.beans.factory.config.AutowireCapableBeanFactory;
28+
import org.springframework.context.support.GenericApplicationContext;
2729
import org.springframework.data.querydsl.QUser;
2830
import org.springframework.data.querydsl.SimpleEntityPathResolver;
2931
import org.springframework.data.querydsl.User;
@@ -33,6 +35,7 @@
3335
import org.springframework.test.util.ReflectionTestUtils;
3436
import org.springframework.web.servlet.ModelAndView;
3537

38+
import com.querydsl.core.types.EntityPath;
3639
import com.querydsl.core.types.Path;
3740
import com.querydsl.core.types.Predicate;
3841

@@ -95,6 +98,27 @@ void shouldReuseExistingQuerydslBinderCustomizer() {
9598
});
9699
}
97100

101+
@Test // #206
102+
@SuppressWarnings({ "unchecked", "rawtypes" })
103+
void shouldApplyDefaultCustomizers() {
104+
105+
GenericApplicationContext context = new GenericApplicationContext();
106+
context.registerBean(DefaultCustomizer.class);
107+
context.refresh();
108+
109+
QuerydslBindingsFactory factory = new QuerydslBindingsFactory(SimpleEntityPathResolver.INSTANCE);
110+
factory.setApplicationContext(context);
111+
112+
QuerydslBindings bindings = factory.createBindingsFor(USER_TYPE, SpecificBinding.class);
113+
Optional<MultiValueBinding<Path<Object>, Object>> binding = bindings
114+
.getBindingForPath(PropertyPathInformation.of("inceptionYear", User.class));
115+
116+
assertThat(binding).hasValueSatisfying(it -> {
117+
Optional<Predicate> bind = it.bind((Path) QUser.user.inceptionYear, Collections.singleton(1L));
118+
assertThat(bind).hasValue(QUser.user.inceptionYear.gt(1L));
119+
});
120+
}
121+
98122
@Test // DATACMNS-669
99123
void rejectsPredicateResolutionIfDomainTypeCantBeAutoDetected() {
100124

@@ -123,4 +147,14 @@ public void customize(QuerydslBindings bindings, QUser user) {
123147
bindings.bind(QUser.user.firstname).firstOptional((path, value) -> value.map(path::contains));
124148
}
125149
}
150+
151+
static class DefaultCustomizer implements DefaultQuerydslBinderCustomizer {
152+
153+
@Override
154+
public void customize(QuerydslBindings bindings, EntityPath<?> root) {
155+
156+
bindings.bind(QUser.user.inceptionYear).first((path, value) -> QUser.user.inceptionYear.gt(value));
157+
}
158+
}
159+
126160
}

0 commit comments

Comments
 (0)