23
23
import org .junit .jupiter .api .Test ;
24
24
import org .junit .jupiter .params .ParameterizedTest ;
25
25
import org .junit .jupiter .params .provider .ValueSource ;
26
-
27
26
import org .springframework .data .jpa .repository .query .QueryRenderer .TokenRenderer ;
28
27
29
28
/**
35
34
*
36
35
* @author Greg Turnquist
37
36
* @author Mark Paluch
37
+ * @author Christoph Strobl
38
38
* @since 3.1
39
39
*/
40
40
class HqlSpecificationTests {
@@ -335,18 +335,15 @@ OR TREAT(e AS Contractor).hours > 100
335
335
""" );
336
336
}
337
337
338
- @ Test // GH-3689
339
- void generic () {
340
-
341
- assertQuery ("""
342
- SELECT e FROM Employee e
343
- WHERE FOO(x).bar RESPECT NULLS
344
- """ );
338
+ @ ParameterizedTest // GH-3689
339
+ @ ValueSource (strings = { "RESPECT NULLS" , "IGNORE NULLS" })
340
+ void generic (String nullHandling ) {
345
341
342
+ // not in the official documentation but supported in the grammar.
346
343
assertQuery ("""
347
344
SELECT e FROM Employee e
348
- WHERE FOO(x).bar IGNORE NULLS
349
- """ );
345
+ WHERE FOO(x).bar %s
346
+ """ . formatted ( nullHandling ) );
350
347
}
351
348
352
349
@ Test // GH-3689
@@ -356,6 +353,11 @@ void size() {
356
353
SELECT e FROM Employee e
357
354
WHERE SIZE(x) > 1
358
355
""" );
356
+
357
+ assertQuery ("""
358
+ SELECT e FROM Employee e
359
+ WHERE SIZE(e.skills) > 1
360
+ """ );
359
361
}
360
362
361
363
@ Test // GH-3689
@@ -384,10 +386,15 @@ WHERE TRUNC(x) = TRUNCATE(y)
384
386
SELECT e FROM Employee e
385
387
WHERE TRUNC(e, 'foo') = TRUNCATE(e, 'bar')
386
388
""" );
389
+
390
+ assertQuery ("""
391
+ SELECT e FROM Employee e
392
+ WHERE TRUNC(e, 'YEAR') = TRUNCATE(LOCAL DATETIME, 'YEAR')
393
+ """ );
387
394
}
388
395
389
396
@ ParameterizedTest // GH-3689
390
- @ ValueSource (strings = { "YYYY " , "MONTH" , "DAY" , "WEEK" , "QUARTER" , "HOUR" , "MINUTE" , "SECOND" , "NANOSECOND" ,
397
+ @ ValueSource (strings = { "YEAR " , "MONTH" , "DAY" , "WEEK" , "QUARTER" , "HOUR" , "MINUTE" , "SECOND" , "NANOSECOND" ,
391
398
"NANOSECOND" , "EPOCH" })
392
399
void trunc (String truncation ) {
393
400
@@ -402,7 +409,17 @@ void format() {
402
409
403
410
assertQuery ("""
404
411
SELECT e FROM Employee e
405
- WHERE FORMAT(x AS 'foo') = FORMAT(x AS 'bar')
412
+ WHERE FORMAT(x AS 'yyyy') = FORMAT(e.hiringDate AS 'yyyy')
413
+ """ );
414
+
415
+ assertQuery ("""
416
+ SELECT e FROM Employee e
417
+ WHERE e.hiringDate = format(LOCAL DATETIME as 'yyyy-MM-dd')
418
+ """ );
419
+
420
+ assertQuery ("""
421
+ SELECT e FROM Employee e
422
+ WHERE e.hiringDate = format(LOCAL_DATE() as 'yyyy-MM-dd')
406
423
""" );
407
424
}
408
425
@@ -411,7 +428,7 @@ void collate() {
411
428
412
429
assertQuery ("""
413
430
SELECT e FROM Employee e
414
- WHERE COLLATE(x AS foo ) = COLLATE(x AS foo.bar )
431
+ WHERE COLLATE(x AS ucs_basic ) = COLLATE(e.name AS ucs_basic )
415
432
""" );
416
433
}
417
434
@@ -424,11 +441,20 @@ void substring() {
424
441
assertQuery ("select substring(c.number, 1) " + //
425
442
"from Call c" );
426
443
444
+ assertQuery ("select substring(c.number, 1, position('/0' in c.number)) " + //
445
+ "from Call c" );
446
+
427
447
assertQuery ("select substring(c.number FROM 1 FOR 2) " + //
428
448
"from Call c" );
429
449
430
450
assertQuery ("select substring(c.number FROM 1) " + //
431
451
"from Call c" );
452
+
453
+ assertQuery ("select substring(c.number FROM 1 FOR position('/0' in c.number)) " + //
454
+ "from Call c" );
455
+
456
+ assertQuery ("select substring(c.number FROM 1) AS shortNumber " + //
457
+ "from Call c" );
432
458
}
433
459
434
460
@ Test // GH-3689
@@ -462,6 +488,9 @@ void position() {
462
488
463
489
assertQuery ("select POSITION(c.number IN 'foo') " + //
464
490
"from Call c " );
491
+
492
+ assertQuery ("select POSITION(c.number IN 'foo') + 1 AS pos " + //
493
+ "from Call c " );
465
494
}
466
495
467
496
@ Test // GH-3689
@@ -490,20 +519,27 @@ void currentDateFunctions() {
490
519
491
520
assertQuery ("select OFFSET DATETIME, OFFSET_DATETIME() " + //
492
521
"from Call c " );
522
+
523
+ assertQuery ("select OFFSET DATETIME AS offsetDatetime, OFFSET_DATETIME() AS offset_datetime " + //
524
+ "from Call c " );
493
525
}
494
526
495
527
@ Test // GH-3689
496
528
void cube () {
497
529
498
530
assertQuery ("select CUBE(foo), CUBE(foo, bar) " + //
499
531
"from Call c " );
532
+
533
+ assertQuery ("select c.callerId from Call c GROUP BY CUBE(state, province)" );
500
534
}
501
535
502
536
@ Test // GH-3689
503
537
void rollup () {
504
538
505
539
assertQuery ("select ROLLUP(foo), ROLLUP(foo, bar) " + //
506
540
"from Call c " );
541
+
542
+ assertQuery ("select c.callerId from Call c GROUP BY ROLLUP(state, province)" );
507
543
}
508
544
509
545
@ Test
@@ -710,16 +746,13 @@ WHERE FUNCTION('hasGoodCredit', c.balance, c.creditLimit) = TRUE
710
746
""" );
711
747
}
712
748
713
- @ Test // GH-3628
714
- void functionInvocationWithIsBoolean () {
715
-
716
- assertQuery ("""
717
- from RoleTmpl where find_in_set(:appId, appIds) is true
718
- """ );
749
+ @ ParameterizedTest // GH-3628
750
+ @ ValueSource (strings = { "is true" , "is not true" , "is false" , "is not false" })
751
+ void functionInvocationWithIsBoolean (String booleanComparison ) {
719
752
720
753
assertQuery ("""
721
- from RoleTmpl where find_in_set(:appId, appIds) is false
722
- """ );
754
+ from RoleTmpl where find_in_set(:appId, appIds) %s
755
+ """ . formatted ( booleanComparison ) );
723
756
}
724
757
725
758
@ Test
@@ -1094,20 +1127,36 @@ void booleanPredicate() {
1094
1127
""" );
1095
1128
}
1096
1129
1097
- @ Test // GH-3628
1098
- void distinctFromPredicate () {
1130
+ @ ParameterizedTest // GH-3628
1131
+ @ ValueSource (strings = { "IS DISTINCT FROM" , "IS NOT DISTINCT FROM" })
1132
+ void distinctFromPredicate (String distinctFrom ) {
1099
1133
1100
1134
assertQuery ("""
1101
1135
SELECT c
1102
1136
FROM Customer c
1103
- WHERE c.orders IS DISTINCT FROM c.payments
1104
- """ );
1137
+ WHERE c.orders %s c.payments
1138
+ """ . formatted ( distinctFrom ) );
1105
1139
1106
1140
assertQuery ("""
1107
1141
SELECT c
1108
1142
FROM Customer c
1109
- WHERE c.orders IS NOT DISTINCT FROM c.payments
1110
- """ );
1143
+ WHERE c.orders %s c.payments
1144
+ """ .formatted (distinctFrom ));
1145
+
1146
+ assertQuery ("""
1147
+ SELECT c
1148
+ FROM Customer c
1149
+ GROUP BY c.lastname
1150
+ HAVING c.orders %s c.payments
1151
+ """ .formatted (distinctFrom ));
1152
+
1153
+ assertQuery ("""
1154
+ SELECT c
1155
+ FROM Customer c
1156
+ WHERE EXISTS (SELECT c2
1157
+ FROM Customer c2
1158
+ WHERE c2.orders %s c.orders)
1159
+ """ .formatted (distinctFrom ));
1111
1160
}
1112
1161
1113
1162
@ Test
@@ -1576,7 +1625,7 @@ void hqlQueries() {
1576
1625
assertQuery ("select longest.duration " + //
1577
1626
"from Phone p " + //
1578
1627
"left join lateral (" + //
1579
- " select c.duration as duration " + //
1628
+ "select c.duration as duration " + //
1580
1629
" from p.calls c" + //
1581
1630
" order by c.duration desc" + //
1582
1631
" limit 1 " + //
0 commit comments