17
17
18
18
import io .cdap .cdap .api .common .Bytes ;
19
19
import io .cdap .cdap .api .data .schema .Schema ;
20
+ import io .cdap .cdap .api .data .schema .Schema .LogicalType ;
20
21
import io .cdap .wrangler .api .DirectiveExecutionException ;
21
22
import io .cdap .wrangler .api .DirectiveParseException ;
22
23
import io .cdap .wrangler .api .Row ;
23
24
24
25
import java .math .BigDecimal ;
26
+ import java .math .MathContext ;
25
27
import java .math .RoundingMode ;
26
28
import java .util .Collections ;
27
29
import java .util .HashMap ;
@@ -45,7 +47,7 @@ private ColumnConverter() {
45
47
* @throws DirectiveExecutionException when a column matching the target name already exists
46
48
*/
47
49
public static void rename (String directiveName , Row row , String column , String toName )
48
- throws DirectiveExecutionException {
50
+ throws DirectiveExecutionException {
49
51
int idx = row .find (column );
50
52
int existingColumn = row .find (toName );
51
53
if (idx == -1 ) {
@@ -57,9 +59,9 @@ public static void rename(String directiveName, Row row, String column, String t
57
59
row .setColumn (idx , toName );
58
60
} else {
59
61
throw new DirectiveExecutionException (
60
- directiveName , String .format ("Column '%s' already exists. Apply the 'drop %s' directive before " +
61
- "renaming '%s' to '%s'." ,
62
- toName , toName , column , toName ));
62
+ directiveName , String .format ("Column '%s' already exists. Apply the 'drop %s' directive before " +
63
+ "renaming '%s' to '%s'." ,
64
+ toName , toName , column , toName ));
63
65
}
64
66
}
65
67
@@ -73,8 +75,8 @@ public static void rename(String directiveName, Row row, String column, String t
73
75
* @throws DirectiveExecutionException when an unsupported type is specified or the column can not be converted.
74
76
*/
75
77
public static void convertType (String directiveName , Row row , String column , String toType ,
76
- Integer scale , RoundingMode roundingMode )
77
- throws DirectiveExecutionException {
78
+ Integer scale , Integer precision , RoundingMode roundingMode )
79
+ throws DirectiveExecutionException {
78
80
int idx = row .find (column );
79
81
if (idx != -1 ) {
80
82
Object object = row .getValue (idx );
@@ -84,21 +86,22 @@ public static void convertType(String directiveName, Row row, String column, Str
84
86
try {
85
87
Object converted = ColumnConverter .convertType (column , toType , object );
86
88
if (toType .equalsIgnoreCase (ColumnTypeNames .DECIMAL )) {
87
- row .setValue (idx , setDecimalScale ((BigDecimal ) converted , scale , roundingMode ));
89
+ row .setValue (idx , setDecimalScaleAndPrecision ((BigDecimal ) converted , scale ,
90
+ precision , roundingMode ));
88
91
} else {
89
92
row .setValue (idx , converted );
90
93
}
91
94
} catch (DirectiveExecutionException e ) {
92
95
throw e ;
93
96
} catch (Exception e ) {
94
97
throw new DirectiveExecutionException (
95
- directiveName , String .format ("Column '%s' cannot be converted to a '%s'." , column , toType ), e );
98
+ directiveName , String .format ("Column '%s' cannot be converted to a '%s'." , column , toType ), e );
96
99
}
97
100
}
98
101
}
99
102
100
103
private static Object convertType (String col , String toType , Object object )
101
- throws Exception {
104
+ throws Exception {
102
105
toType = toType .toUpperCase ();
103
106
switch (toType ) {
104
107
case ColumnTypeNames .INTEGER :
@@ -291,38 +294,62 @@ private static Object convertType(String col, String toType, Object object)
291
294
292
295
default :
293
296
throw new DirectiveExecutionException (String .format (
294
- "Column '%s' is of unsupported type '%s'. Supported types are: " +
295
- "int, short, long, double, decimal, boolean, string, bytes" , col , toType ));
297
+ "Column '%s' is of unsupported type '%s'. Supported types are: " +
298
+ "int, short, long, double, decimal, boolean, string, bytes" , col , toType ));
296
299
}
297
300
throw new DirectiveExecutionException (
298
301
String .format ("Column '%s' has value of type '%s' and cannot be converted to a '%s'." , col ,
299
302
object .getClass ().getSimpleName (), toType ));
300
303
}
301
304
302
- private static BigDecimal setDecimalScale (BigDecimal decimal , Integer scale , RoundingMode roundingMode )
303
- throws DirectiveExecutionException {
304
- if (scale == null ) {
305
+ private static BigDecimal setDecimalScaleAndPrecision (BigDecimal decimal , Integer scale ,
306
+ Integer precision , RoundingMode roundingMode )
307
+ throws DirectiveExecutionException {
308
+ if (scale == null && precision == null ) {
305
309
return decimal ;
306
310
}
307
311
try {
308
- return decimal .setScale (scale , roundingMode );
312
+ if (precision == null ) {
313
+ return decimal .setScale (scale , roundingMode );
314
+ } else if (scale == null ) {
315
+ return decimal .round (new MathContext (precision , roundingMode ));
316
+ } else {
317
+ BigDecimal result ;
318
+ if (validateScaleAndPrecision (scale , precision , decimal )) {
319
+ result = decimal .setScale (scale , roundingMode );
320
+ result = result .round (new MathContext (precision , roundingMode ));
321
+ } else {
322
+ throw new DirectiveExecutionException (String .format (
323
+ "Cannot set scale as '%s' and precision as '%s' for value '%s' when"
324
+ + "given precision - scale is less than number of digits"
325
+ + " before decimal point " , scale , precision , decimal ));
326
+ }
327
+ return result ;
328
+ }
309
329
} catch (ArithmeticException e ) {
310
330
throw new DirectiveExecutionException (String .format (
311
- "Cannot set scale as '%s' for value '%s' when rounding-mode is '%s'" , scale , decimal , roundingMode ), e );
331
+ "Cannot set scale as '%s' and precision '%s' for value '%s' when rounding-mode "
332
+ + "is '%s'" , scale , precision , decimal , roundingMode ), e );
312
333
}
313
334
}
314
335
315
- public static Schema getSchemaForType (String type , Integer scale ) throws DirectiveParseException {
336
+ private static Boolean validateScaleAndPrecision (Integer scale , Integer precision , BigDecimal decimal ) {
337
+ int digitsBeforeDecimalPoint = decimal .signum () == 0 ? 1 : decimal .precision () - decimal .scale ();
338
+ return precision - scale >= digitsBeforeDecimalPoint ;
339
+ }
340
+
341
+ public static Schema getSchemaForType (String type , Integer scale , Integer precision ) throws DirectiveParseException {
316
342
Schema typeSchema ;
317
343
type = type .toUpperCase ();
318
344
if (type .equals (ColumnTypeNames .DECIMAL )) {
319
345
// TODO make set-type support setting decimal precision
346
+ precision = precision != null ? precision : 77 ;
320
347
scale = scale != null ? scale : 38 ;
321
- typeSchema = Schema .nullableOf (Schema .decimalOf (77 , scale ));
348
+ typeSchema = Schema .nullableOf (Schema .decimalOf (precision , scale ));
322
349
} else {
323
350
if (!SCHEMA_TYPE_MAP .containsKey (type )) {
324
351
throw new DirectiveParseException (String .format ("'%s' is an unsupported type. " +
325
- "Supported types are: int, short, long, double, decimal, boolean, string, bytes" , type ));
352
+ "Supported types are: int, short, long, double, decimal, boolean, string, bytes" , type ));
326
353
}
327
354
typeSchema = Schema .nullableOf (Schema .of (SCHEMA_TYPE_MAP .get (type )));
328
355
}
0 commit comments