13
13
import static org .opensearch .sql .data .model .ExprValueUtils .LITERAL_NULL ;
14
14
import static org .opensearch .sql .data .model .ExprValueUtils .LITERAL_TRUE ;
15
15
16
+ import java .util .Arrays ;
16
17
import java .util .LinkedHashMap ;
17
18
import java .util .List ;
18
19
import java .util .Map ;
19
20
import org .junit .jupiter .api .Test ;
20
21
import org .junit .jupiter .api .extension .ExtendWith ;
22
+ import org .junit .jupiter .api .extension .TestInstantiationException ;
21
23
import org .mockito .junit .jupiter .MockitoExtension ;
22
24
import org .opensearch .sql .data .model .ExprBooleanValue ;
23
25
import org .opensearch .sql .data .model .ExprCollectionValue ;
@@ -234,7 +236,7 @@ void json_returnsSemanticCheckException() {
234
236
@ Test
235
237
void json_extract_search () {
236
238
ExprValue expected = new ExprIntegerValue (1 );
237
- execute_extract_json (expected , "{\" a\" :1}" , "$.a" );
239
+ assert_equals_extract_json (expected , "{\" a\" :1}" , "$.a" );
238
240
}
239
241
240
242
@ Test
@@ -256,17 +258,17 @@ void json_extract_search_arrays() {
256
258
// extract specific index from JSON list
257
259
for (int i = 0 ; i < expectedExprValues .size (); i ++) {
258
260
String path = String .format ("$.a[%d]" , i );
259
- execute_extract_json (expectedExprValues .get (i ), jsonArray , path );
261
+ assert_equals_extract_json (expectedExprValues .get (i ), jsonArray , path );
260
262
}
261
263
262
264
// extract nested object
263
265
ExprValue nestedExpected =
264
266
ExprTupleValue .fromExprValueMap (Map .of ("d" , new ExprIntegerValue (1 )));
265
- execute_extract_json (nestedExpected , jsonArray , "$.a[5].c" );
267
+ assert_equals_extract_json (nestedExpected , jsonArray , "$.a[5].c" );
266
268
267
269
// extract * from JSON list
268
270
ExprValue starExpected = new ExprCollectionValue (expectedExprValues );
269
- execute_extract_json (starExpected , jsonArray , "$.a[*]" );
271
+ assert_equals_extract_json (starExpected , jsonArray , "$.a[*]" );
270
272
}
271
273
272
274
@ Test
@@ -285,10 +287,11 @@ void json_extract_returns_null() {
285
287
"false" ,
286
288
"" );
287
289
288
- jsonStrings .forEach (str -> execute_extract_json (LITERAL_NULL , str , "$.a.path_not_found_key" ));
290
+ jsonStrings .forEach (
291
+ str -> assert_equals_extract_json (LITERAL_NULL , str , "$.a.path_not_found_key" ));
289
292
290
293
// null string literal
291
- assertEquals (LITERAL_NULL , DSL . jsonExtract ( DSL . literal ( "null" ), DSL . literal ( "$.a" )). valueOf () );
294
+ assert_equals_extract_json (LITERAL_NULL , "null" , "$.a" );
292
295
293
296
// null json
294
297
assertEquals (
@@ -300,7 +303,7 @@ void json_extract_returns_null() {
300
303
DSL .jsonExtract (DSL .literal (LITERAL_MISSING ), DSL .literal ("$.a" )).valueOf ());
301
304
302
305
// array out of bounds
303
- execute_extract_json (LITERAL_NULL , "{\" a\" :[1,2,3]}" , "$.a[4]" );
306
+ assert_equals_extract_json (LITERAL_NULL , "{\" a\" :[1,2,3]}" , "$.a[4]" );
304
307
}
305
308
306
309
@ Test
@@ -334,14 +337,10 @@ void json_extract_throws_SemanticCheckException() {
334
337
@ Test
335
338
void json_extract_throws_ExpressionEvaluationException () {
336
339
// null path
337
- assertThrows (
338
- ExpressionEvaluationException .class ,
339
- () -> DSL .jsonExtract (DSL .literal ("{\" a\" :1}" ), DSL .literal (LITERAL_NULL )).valueOf ());
340
+ assert_throws_extract_json (ExpressionEvaluationException .class , "{\" a\" :1}" , LITERAL_NULL );
340
341
341
342
// missing path
342
- assertThrows (
343
- ExpressionEvaluationException .class ,
344
- () -> DSL .jsonExtract (DSL .literal ("{\" a\" :1}" ), DSL .literal (LITERAL_MISSING )).valueOf ());
343
+ assert_throws_extract_json (ExpressionEvaluationException .class , "{\" a\" :1}" , LITERAL_MISSING );
345
344
}
346
345
347
346
@ Test
@@ -350,21 +349,59 @@ void json_extract_search_list_of_paths() {
350
349
"{\" foo\" : \" foo\" , \" fuzz\" : true, \" bar\" : 1234, \" bar2\" : 12.34, \" baz\" : null, "
351
350
+ "\" obj\" : {\" internal\" : \" value\" }, \" arr\" : [\" string\" , true, null]}" ;
352
351
353
- ExprValue expected =
352
+ // scalar results with one invalid path
353
+ ExprValue expected_scalar_results =
354
354
new ExprCollectionValue (
355
355
List .of (new ExprStringValue ("foo" ), new ExprFloatValue (12.34 ), LITERAL_NULL ));
356
- Expression pathExpr1 = DSL .literal (ExprValueUtils .stringValue ("$.foo" ));
357
- Expression pathExpr2 = DSL .literal (ExprValueUtils .stringValue ("$.bar2" ));
358
- Expression pathExpr3 = DSL .literal (ExprValueUtils .stringValue ("$.potato" ));
359
- Expression jsonExpr = DSL .literal (ExprValueUtils .stringValue (objectJson ));
360
- ExprValue actual = DSL .jsonExtract (jsonExpr , pathExpr1 , pathExpr2 , pathExpr3 ).valueOf ();
361
- assertEquals (expected , actual );
356
+
357
+ assert_equals_extract_json (expected_scalar_results , objectJson , "$.foo" , "$.bar2" , "$.potato" );
358
+
359
+ ExprValue expected_multivalued_results =
360
+ new ExprCollectionValue (
361
+ List .of (
362
+ new ExprCollectionValue (
363
+ List .of (new ExprStringValue ("string" ), LITERAL_TRUE , LITERAL_NULL )),
364
+ ExprTupleValue .fromExprValueMap (Map .of ("internal" , new ExprStringValue ("value" ))),
365
+ new ExprFloatValue (12.34 )));
366
+
367
+ // path returns array and struct
368
+ assert_equals_extract_json (
369
+ expected_multivalued_results , objectJson , "$.arr" , "$.obj" , "$.bar2" );
370
+
371
+ // path returns multivalued result
372
+ assert_equals_extract_json (
373
+ expected_multivalued_results , objectJson , "$.arr[*]" , "$.obj" , "$.bar2" );
362
374
}
363
375
364
- private static void execute_extract_json (ExprValue expected , String json , String path ) {
365
- Expression pathExpr = DSL .literal (ExprValueUtils .stringValue (path ));
366
- Expression jsonExpr = DSL .literal (ExprValueUtils .stringValue (json ));
367
- ExprValue actual = DSL .jsonExtract (jsonExpr , pathExpr ).valueOf ();
376
+ private static void assert_equals_extract_json (ExprValue expected , Object json , Object ... paths ) {
377
+ ExprValue actual = execute_extract_json (json , paths );
368
378
assertEquals (expected , actual );
369
379
}
380
+
381
+ private static <T extends Throwable > void assert_throws_extract_json (
382
+ Class <T > expectedError , Object json , Object ... paths ) {
383
+ assertThrows (expectedError , () -> execute_extract_json (json , paths ));
384
+ }
385
+
386
+ private static ExprValue execute_extract_json (Object json , Object [] paths ) {
387
+ Expression jsonExpr = object_to_expr (json );
388
+ List <Expression > pathExpressions =
389
+ Arrays .stream (paths ).map (JsonFunctionsTest ::object_to_expr ).toList ();
390
+
391
+ return switch (paths .length ) {
392
+ case 1 -> DSL .jsonExtract (jsonExpr , pathExpressions .getFirst ()).valueOf ();
393
+ case 2 -> DSL .jsonExtract (jsonExpr , pathExpressions .getFirst (), pathExpressions .get (1 ))
394
+ .valueOf ();
395
+ case 3 -> DSL .jsonExtract (
396
+ jsonExpr , pathExpressions .getFirst (), pathExpressions .get (1 ), pathExpressions .get (2 ))
397
+ .valueOf ();
398
+ default -> throw new TestInstantiationException ("Invalid number of paths provided." );
399
+ };
400
+ }
401
+
402
+ private static Expression object_to_expr (Object val ) {
403
+ return (val instanceof String )
404
+ ? DSL .literal (ExprValueUtils .stringValue ((String ) val ))
405
+ : DSL .literal ((ExprValue ) val );
406
+ }
370
407
}
0 commit comments