20
20
21
21
import java .util .Objects ;
22
22
23
- import org .springframework .data .domain .Pageable ;
24
-
25
23
import org .jspecify .annotations .Nullable ;
24
+
25
+ import org .springframework .data .domain .Pageable ;
26
26
import org .springframework .data .domain .Sort ;
27
27
import org .springframework .data .expression .ValueEvaluationContextProvider ;
28
28
import org .springframework .data .jpa .repository .QueryRewriter ;
32
32
import org .springframework .data .util .Lazy ;
33
33
import org .springframework .util .Assert ;
34
34
import org .springframework .util .ConcurrentLruCache ;
35
- import org .springframework .util .StringUtils ;
36
35
37
36
/**
38
37
* Base class for {@link String} based JPA queries.
49
48
*/
50
49
abstract class AbstractStringBasedJpaQuery extends AbstractJpaQuery {
51
50
52
- private final StringQuery query ;
53
- private final Lazy <IntrospectedQuery > countQuery ;
51
+ private final EntityQuery query ;
52
+ private final Lazy <ParametrizedQuery > countQuery ;
54
53
private final ValueExpressionDelegate valueExpressionDelegate ;
55
54
private final QueryRewriter queryRewriter ;
56
55
private final QuerySortRewriter querySortRewriter ;
@@ -64,25 +63,42 @@ abstract class AbstractStringBasedJpaQuery extends AbstractJpaQuery {
64
63
* @param method must not be {@literal null}.
65
64
* @param em must not be {@literal null}.
66
65
* @param queryString must not be {@literal null}.
67
- * @param countQueryString must not be {@literal null}.
66
+ * @param countQuery can be {@literal null} if not defined .
68
67
* @param queryConfiguration must not be {@literal null}.
69
68
*/
70
- public AbstractStringBasedJpaQuery (JpaQueryMethod method , EntityManager em , String queryString ,
69
+ AbstractStringBasedJpaQuery (JpaQueryMethod method , EntityManager em , String queryString ,
71
70
@ Nullable String countQueryString , JpaQueryConfiguration queryConfiguration ) {
71
+ this (method , em , method .getDeclaredQuery (queryString ),
72
+ countQueryString != null ? method .getDeclaredQuery (countQueryString ) : null , queryConfiguration );
73
+ }
74
+
75
+ /**
76
+ * Creates a new {@link AbstractStringBasedJpaQuery} from the given {@link JpaQueryMethod}, {@link EntityManager} and
77
+ * query {@link String}.
78
+ *
79
+ * @param method must not be {@literal null}.
80
+ * @param em must not be {@literal null}.
81
+ * @param query must not be {@literal null}.
82
+ * @param countQuery can be {@literal null}.
83
+ * @param queryConfiguration must not be {@literal null}.
84
+ */
85
+ public AbstractStringBasedJpaQuery (JpaQueryMethod method , EntityManager em , DeclaredQuery query ,
86
+ @ Nullable DeclaredQuery countQuery , JpaQueryConfiguration queryConfiguration ) {
72
87
73
88
super (method , em );
74
89
75
- Assert .hasText ( queryString , "Query string must not be null or empty " );
90
+ Assert .notNull ( query , "Query must not be null" );
76
91
Assert .notNull (queryConfiguration , "JpaQueryConfiguration must not be null" );
77
92
78
93
this .valueExpressionDelegate = queryConfiguration .getValueExpressionDelegate ();
79
94
this .valueExpressionContextProvider = valueExpressionDelegate .createValueContextProvider (method .getParameters ());
80
- this .query = ExpressionBasedStringQuery .create (queryString , method , queryConfiguration );
95
+
96
+ this .query = TemplatedQuery .create (query , method .getEntityInformation (), queryConfiguration );
81
97
82
98
this .countQuery = Lazy .of (() -> {
83
99
84
- if (StringUtils . hasText ( countQueryString ) ) {
85
- return ExpressionBasedStringQuery .create (countQueryString , method , queryConfiguration );
100
+ if (countQuery != null ) {
101
+ return TemplatedQuery .create (countQuery , method . getEntityInformation () , queryConfiguration );
86
102
}
87
103
88
104
return this .query .deriveCountQuery (method .getCountQueryProjection ());
@@ -108,21 +124,25 @@ public AbstractStringBasedJpaQuery(JpaQueryMethod method, EntityManager em, Stri
108
124
"JDBC style parameters (?) are not supported for JPA queries" );
109
125
}
110
126
127
+ private DeclaredQuery createQuery (String queryString , boolean nativeQuery ) {
128
+ return nativeQuery ? DeclaredQuery .nativeQuery (queryString ) : DeclaredQuery .jpqlQuery (queryString );
129
+ }
130
+
111
131
@ Override
112
132
public Query doCreateQuery (JpaParametersParameterAccessor accessor ) {
113
133
114
134
Sort sort = accessor .getSort ();
115
135
ResultProcessor processor = getQueryMethod ().getResultProcessor ().withDynamicProjection (accessor );
116
136
ReturnedType returnedType = processor .getReturnedType ();
117
- String sortedQueryString = getSortedQueryString (sort , returnedType );
118
- Query query = createJpaQuery (sortedQueryString , sort , accessor .getPageable (), returnedType );
137
+ QueryProvider sortedQuery = getSortedQuery (sort , returnedType );
138
+ Query query = createJpaQuery (sortedQuery , sort , accessor .getPageable (), returnedType );
119
139
120
140
// it is ok to reuse the binding contained in the ParameterBinder, although we create a new query String because the
121
141
// parameters in the query do not change.
122
142
return parameterBinder .get ().bindAndPrepare (query , accessor );
123
143
}
124
144
125
- String getSortedQueryString (Sort sort , ReturnedType returnedType ) {
145
+ QueryProvider getSortedQuery (Sort sort , ReturnedType returnedType ) {
126
146
return querySortRewriter .getSorted (query , sort , returnedType );
127
147
}
128
148
@@ -131,7 +151,7 @@ protected ParameterBinder createBinder() {
131
151
return createBinder (query );
132
152
}
133
153
134
- protected ParameterBinder createBinder (IntrospectedQuery query ) {
154
+ protected ParameterBinder createBinder (ParametrizedQuery query ) {
135
155
return ParameterBinderFactory .createQueryAwareBinder (getQueryMethod ().getParameters (), query ,
136
156
valueExpressionDelegate , valueExpressionContextProvider );
137
157
}
@@ -162,28 +182,28 @@ public EntityQuery getQuery() {
162
182
/**
163
183
* @return the countQuery
164
184
*/
165
- public IntrospectedQuery getCountQuery () {
185
+ public ParametrizedQuery getCountQuery () {
166
186
return countQuery .get ();
167
187
}
168
188
169
189
/**
170
190
* Creates an appropriate JPA query from an {@link EntityManager} according to the current {@link AbstractJpaQuery}
171
191
* type.
172
192
*/
173
- protected Query createJpaQuery (String queryString , Sort sort , @ Nullable Pageable pageable ,
193
+ protected Query createJpaQuery (QueryProvider query , Sort sort , @ Nullable Pageable pageable ,
174
194
ReturnedType returnedType ) {
175
195
176
196
EntityManager em = getEntityManager ();
177
197
178
198
if (this .query .hasConstructorExpression () || this .query .isDefaultProjection ()) {
179
- return em .createQuery (potentiallyRewriteQuery (queryString , sort , pageable ));
199
+ return em .createQuery (potentiallyRewriteQuery (query . getQueryString () , sort , pageable ));
180
200
}
181
201
182
202
Class <?> typeToRead = getTypeToRead (returnedType );
183
203
184
204
return typeToRead == null //
185
- ? em .createQuery (potentiallyRewriteQuery (queryString , sort , pageable )) //
186
- : em .createQuery (potentiallyRewriteQuery (queryString , sort , pageable ), typeToRead );
205
+ ? em .createQuery (potentiallyRewriteQuery (query . getQueryString () , sort , pageable )) //
206
+ : em .createQuery (potentiallyRewriteQuery (query . getQueryString () , sort , pageable ), typeToRead );
187
207
}
188
208
189
209
/**
@@ -202,16 +222,16 @@ protected String potentiallyRewriteQuery(String originalQuery, Sort sort, @Nulla
202
222
: queryRewriter .rewrite (originalQuery , sort );
203
223
}
204
224
205
- String applySorting (CachableQuery cachableQuery ) {
206
- return cachableQuery .getDeclaredQuery (). getQueryEnhancer ()
225
+ QueryProvider applySorting (CachableQuery cachableQuery ) {
226
+ return cachableQuery .getDeclaredQuery ()
207
227
.rewrite (new DefaultQueryRewriteInformation (cachableQuery .getSort (), cachableQuery .getReturnedType ()));
208
228
}
209
229
210
230
/**
211
231
* Query Sort Rewriter interface.
212
232
*/
213
233
interface QuerySortRewriter {
214
- String getSorted (StringQuery query , Sort sort , ReturnedType returnedType );
234
+ QueryProvider getSorted (EntityQuery query , Sort sort , ReturnedType returnedType );
215
235
}
216
236
217
237
/**
@@ -221,28 +241,28 @@ enum SimpleQuerySortRewriter implements QuerySortRewriter {
221
241
222
242
INSTANCE ;
223
243
224
- public String getSorted (StringQuery query , Sort sort , ReturnedType returnedType ) {
225
- return query .getQueryEnhancer (). rewrite (new DefaultQueryRewriteInformation (sort , returnedType ));
244
+ public QueryProvider getSorted (EntityQuery query , Sort sort , ReturnedType returnedType ) {
245
+ return query .rewrite (new DefaultQueryRewriteInformation (sort , returnedType ));
226
246
}
227
247
}
228
248
229
249
static class UnsortedCachingQuerySortRewriter implements QuerySortRewriter {
230
250
231
- private volatile @ Nullable String cachedQueryString ;
251
+ private volatile @ Nullable QueryProvider cachedQuery ;
232
252
233
- public String getSorted (StringQuery query , Sort sort , ReturnedType returnedType ) {
253
+ public QueryProvider getSorted (EntityQuery query , Sort sort , ReturnedType returnedType ) {
234
254
235
255
if (sort .isSorted ()) {
236
256
throw new UnsupportedOperationException ("NoOpQueryCache does not support sorting" );
237
257
}
238
258
239
- String cachedQueryString = this .cachedQueryString ;
240
- if (cachedQueryString == null ) {
241
- this .cachedQueryString = cachedQueryString = query . getQueryEnhancer ()
259
+ QueryProvider cachedQuery = this .cachedQuery ;
260
+ if (cachedQuery == null ) {
261
+ this .cachedQuery = cachedQuery = query
242
262
.rewrite (new DefaultQueryRewriteInformation (sort , returnedType ));
243
263
}
244
264
245
- return cachedQueryString ;
265
+ return cachedQuery ;
246
266
}
247
267
}
248
268
@@ -251,22 +271,22 @@ public String getSorted(StringQuery query, Sort sort, ReturnedType returnedType)
251
271
*/
252
272
class CachingQuerySortRewriter implements QuerySortRewriter {
253
273
254
- private final ConcurrentLruCache <CachableQuery , String > queryCache = new ConcurrentLruCache <>(16 ,
274
+ private final ConcurrentLruCache <CachableQuery , QueryProvider > queryCache = new ConcurrentLruCache <>(16 ,
255
275
AbstractStringBasedJpaQuery .this ::applySorting );
256
276
257
- private volatile @ Nullable String cachedQueryString ;
277
+ private volatile @ Nullable QueryProvider cachedQuery ;
258
278
259
279
@ Override
260
- public String getSorted (StringQuery query , Sort sort , ReturnedType returnedType ) {
280
+ public QueryProvider getSorted (EntityQuery query , Sort sort , ReturnedType returnedType ) {
261
281
262
282
if (sort .isUnsorted ()) {
263
283
264
- String cachedQueryString = this .cachedQueryString ;
265
- if (cachedQueryString == null ) {
266
- this .cachedQueryString = cachedQueryString = queryCache .get (new CachableQuery (query , sort , returnedType ));
284
+ QueryProvider cachedQuery = this .cachedQuery ;
285
+ if (cachedQuery == null ) {
286
+ this .cachedQuery = cachedQuery = queryCache .get (new CachableQuery (query , sort , returnedType ));
267
287
}
268
288
269
- return cachedQueryString ;
289
+ return cachedQuery ;
270
290
}
271
291
272
292
return queryCache .get (new CachableQuery (query , sort , returnedType ));
@@ -282,20 +302,20 @@ public String getSorted(StringQuery query, Sort sort, ReturnedType returnedType)
282
302
*/
283
303
static class CachableQuery {
284
304
285
- private final StringQuery query ;
305
+ private final EntityQuery query ;
286
306
private final String queryString ;
287
307
private final Sort sort ;
288
308
private final ReturnedType returnedType ;
289
309
290
- CachableQuery (StringQuery query , Sort sort , ReturnedType returnedType ) {
310
+ CachableQuery (EntityQuery query , Sort sort , ReturnedType returnedType ) {
291
311
292
312
this .query = query ;
293
313
this .queryString = query .getQueryString ();
294
314
this .sort = sort ;
295
315
this .returnedType = returnedType ;
296
316
}
297
317
298
- StringQuery getDeclaredQuery () {
318
+ EntityQuery getDeclaredQuery () {
299
319
return query ;
300
320
}
301
321
0 commit comments