Overview
Given
- Spring Data JPA, Spring Data Rest, QueryDsl
- a
Meetupentity- with a
Map<String,String> propertiesfield- persisted in a
MEETUP_PROPERTYtable as an@ElementCollection
- persisted in a
- with a
- a
MeetupRepository- that extends
QueryDslPredicateExecutor<Meetup>
- that extends
I'd expect
A web query of
GET /api/meetup?properties[aKey]=aValue
to return only Meetups with a property entry that has the specified key and value: aKey=aValue.
However, that's not working for me. What am I missing?
Tried
Simple Fields
Simple fields work, like name and description:
GET /api/meetup?name=whatever
Collection fields work, like participants:
GET /api/meetup?participants.name=whatever
But not this Map field.
Customize QueryDsl bindings
I've tried customizing the binding by having the repository
extend QuerydslBinderCustomizer<QMeetup>
and overriding the
customize(QuerydslBindings bindings, QMeetup meetup)
method, but while the customize() method is being hit, the binding code inside the lambda is not.
EDIT: Learned that's because QuerydslBindings means of evaluating the query parameter do not let it match up against the pathSpecs map it's internally holding - which has your custom bindings in it.
Some Specifics
Meetup.properties field
@ElementCollection(fetch = FetchType.EAGER)
@CollectionTable(name = "MEETUP_PROPERTY", joinColumns = @JoinColumn(name = "MEETUP_ID"))
@MapKeyColumn(name = "KEY")
@Column(name = "VALUE", length = 2048)
private Map<String, String> properties = new HashMap<>();
customized querydsl binding
EDIT: See above; turns out, this was doing nothing for my code.
public interface MeetupRepository extends PagingAndSortingRepository<Meetup, Long>,
QueryDslPredicateExecutor<Meetup>,
QuerydslBinderCustomizer<QMeetup> {
@Override
default void customize(QuerydslBindings bindings, QMeetup meetup) {
bindings.bind(meetup.properties).first((path, value) -> {
BooleanBuilder builder = new BooleanBuilder();
for (String key : value.keySet()) {
builder.and(path.containsKey(key).and(path.get(key).eq(value.get(key))));
}
return builder;
});
}
Additional Findings
QuerydslPredicateBuilder.getPredicate()asksQuerydslBindings.getPropertyPath()to try 2 ways to return a path from so it can make a predicate thatQuerydslAwareRootResourceInformationHandlerMethodArgumentResolver.postProcess()can use.- 1 is to look in the customized bindings. I don't see any way to express a map query there
- 2 is to default to Spring's bean paths. Same expression problem there. How do you express a map?
So it looks impossible to get
QuerydslPredicateBuilder.getPredicate()to automatically create a predicate. Fine - I can do it manually, if I can hook intoQuerydslAwareRootResourceInformationHandlerMethodArgumentResolver.postProcess()
HOW can I override that class, or replace the bean? It's instantiated and returned as a bean in the RepositoryRestMvcConfiguration.repoRequestArgumentResolver() bean declaration.
- I can override that bean by declaring my own
repoRequestArgumentResolverbean, but it doesn't get used.- It gets overridden by
RepositoryRestMvcConfigurations. I can't force it by setting it@Primaryor@Ordered(HIGHEST_PRECEDENCE). - I can force it by explicitly component-scanning
RepositoryRestMvcConfiguration.class, but that also messes up Spring Boot's autoconfiguration because it causesRepositoryRestMvcConfiguration'sbean declarations to be processed before any auto-configuration runs. Among other things, that results in responses that are serialized by Jackson in unwanted ways.
- It gets overridden by
The Question
Well - looks like the support I expected just isn't there.
So the question becomes:
HOW do I correctly override the repoRequestArgumentResolver bean?
BTW - QuerydslAwareRootResourceInformationHandlerMethodArgumentResolver is awkwardly non-public. :/