From b165b25772b2d166377c2eab2307f4bc72304096 Mon Sep 17 00:00:00 2001 From: Alec Huang Date: Wed, 31 Jul 2024 15:47:32 -0700 Subject: [PATCH] Add IcebergParquetValueParserTest for primitive columns --- .../IcebergParquetValueParserTest.java | 330 ++++++++++++++++++ .../ParquetValueParserAssertionBuilder.java | 104 ++++++ .../SnowflakeParquetValueParserTest.java | 98 +----- 3 files changed, 438 insertions(+), 94 deletions(-) create mode 100644 src/test/java/net/snowflake/ingest/streaming/internal/IcebergParquetValueParserTest.java create mode 100644 src/test/java/net/snowflake/ingest/streaming/internal/ParquetValueParserAssertionBuilder.java diff --git a/src/test/java/net/snowflake/ingest/streaming/internal/IcebergParquetValueParserTest.java b/src/test/java/net/snowflake/ingest/streaming/internal/IcebergParquetValueParserTest.java new file mode 100644 index 000000000..1b2c4bf52 --- /dev/null +++ b/src/test/java/net/snowflake/ingest/streaming/internal/IcebergParquetValueParserTest.java @@ -0,0 +1,330 @@ +/* + * Copyright (c) 2024 Snowflake Computing Inc. All rights reserved. + */ + +package net.snowflake.ingest.streaming.internal; + +import static java.time.ZoneOffset.UTC; +import static net.snowflake.ingest.streaming.internal.ParquetBufferValue.BYTE_ARRAY_LENGTH_ENCODING_BYTE_LEN; +import static net.snowflake.ingest.streaming.internal.ParquetBufferValue.DEFINITION_LEVEL_ENCODING_BYTE_LEN; + +import java.math.BigDecimal; +import java.math.BigInteger; +import org.apache.parquet.schema.LogicalTypeAnnotation; +import org.apache.parquet.schema.PrimitiveType.PrimitiveTypeName; +import org.apache.parquet.schema.Type; +import org.apache.parquet.schema.Type.Repetition; +import org.apache.parquet.schema.Types; +import org.junit.Test; + +public class IcebergParquetValueParserTest { + + @Test + public void parseValueBoolean() { + Type type = + Types.primitive(PrimitiveTypeName.BOOLEAN, Repetition.OPTIONAL).named("BOOLEAN_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("BOOLEAN_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + true, type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Boolean.class) + .expectedParsedValue(true) + .expectedSize(ParquetBufferValue.BIT_ENCODING_BYTE_LEN + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(BigInteger.valueOf(1)) + .assertMatches(); + } + + @Test + public void parseValueInt() { + Type type = Types.primitive(PrimitiveTypeName.INT32, Repetition.OPTIONAL).named("INT_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("INT_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + Integer.MAX_VALUE, type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Integer.class) + .expectedParsedValue(Integer.MAX_VALUE) + .expectedSize(4.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(BigInteger.valueOf(Integer.MAX_VALUE)) + .assertMatches(); + } + + @Test + public void parseValueDecimalToInt() { + Type type = + Types.primitive(PrimitiveTypeName.INT32, Repetition.OPTIONAL) + .as(LogicalTypeAnnotation.decimalType(4, 9)) + .named("DECIMAL_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("DECIMAL_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + new BigDecimal("12345.6789"), type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Integer.class) + .expectedParsedValue(123456789) + .expectedSize(4.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(BigInteger.valueOf(123456789)) + .assertMatches(); + } + + @Test + public void parseValueDateToInt() { + Type type = + Types.primitive(PrimitiveTypeName.INT32, Repetition.OPTIONAL) + .as(LogicalTypeAnnotation.dateType()) + .named("DATE_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("DATE_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + "2024-01-01", type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Integer.class) + .expectedParsedValue(19723) + .expectedSize(4.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(BigInteger.valueOf(19723)) + .assertMatches(); + } + + @Test + public void parseValueLong() { + Type type = Types.primitive(PrimitiveTypeName.INT64, Repetition.OPTIONAL).named("LONG_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("LONG_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + Long.MAX_VALUE, type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Long.class) + .expectedParsedValue(Long.MAX_VALUE) + .expectedSize(8.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(BigInteger.valueOf(Long.MAX_VALUE)) + .assertMatches(); + } + + @Test + public void parseValueDecimalToLong() { + Type type = + Types.primitive(PrimitiveTypeName.INT64, Repetition.OPTIONAL) + .as(LogicalTypeAnnotation.decimalType(9, 18)) + .named("DECIMAL_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("DECIMAL_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + new BigDecimal("123456789.123456789"), type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Long.class) + .expectedParsedValue(123456789123456789L) + .expectedSize(8.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(BigInteger.valueOf(123456789123456789L)) + .assertMatches(); + } + + @Test + public void parseValueTimeToLong() { + Type type = + Types.primitive(PrimitiveTypeName.INT64, Repetition.OPTIONAL) + .as(LogicalTypeAnnotation.timeType(false, LogicalTypeAnnotation.TimeUnit.MICROS)) + .named("TIME_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("TIME_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + "12:34:56.789", type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Long.class) + .expectedParsedValue(45296789000L) + .expectedSize(8.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(BigInteger.valueOf(45296789000L)) + .assertMatches(); + } + + @Test + public void parseValueTimestampToLong() { + Type type = + Types.primitive(PrimitiveTypeName.INT64, Repetition.OPTIONAL) + .as(LogicalTypeAnnotation.timestampType(false, LogicalTypeAnnotation.TimeUnit.MICROS)) + .named("TIMESTAMP_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("TIMESTAMP_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + "2024-01-01T12:34:56.789+08:00", type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Long.class) + .expectedParsedValue(1704112496789000L) + .expectedSize(8.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(BigInteger.valueOf(1704112496789000L)) + .assertMatches(); + } + + @Test + public void parseValueTimestampTZToLong() { + Type type = + Types.primitive(PrimitiveTypeName.INT64, Repetition.OPTIONAL) + .as(LogicalTypeAnnotation.timestampType(true, LogicalTypeAnnotation.TimeUnit.MICROS)) + .named("TIMESTAMP_TZ_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("TIMESTAMP_TZ_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + "2024-01-01T12:34:56.789+08:00", type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Long.class) + .expectedParsedValue(1704083696789000L) + .expectedSize(8.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(BigInteger.valueOf(1704083696789000L)) + .assertMatches(); + } + + @Test + public void parseValueFloat() { + Type type = Types.primitive(PrimitiveTypeName.FLOAT, Repetition.OPTIONAL).named("FLOAT_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("FLOAT_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + Float.MAX_VALUE, type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Float.class) + .expectedParsedValue(Float.MAX_VALUE) + .expectedSize(4.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax((double) Float.MAX_VALUE) + .assertMatches(); + } + + @Test + public void parseValueDouble() { + Type type = Types.primitive(PrimitiveTypeName.DOUBLE, Repetition.OPTIONAL).named("DOUBLE_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("DOUBLE_COL"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + Double.MAX_VALUE, type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(Double.class) + .expectedParsedValue(Double.MAX_VALUE) + .expectedSize(8.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN) + .expectedMinMax(Double.MAX_VALUE) + .assertMatches(); + } + + @Test + public void parseValueBinary() { + Type type = Types.primitive(PrimitiveTypeName.BINARY, Repetition.OPTIONAL).named("BINARY_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("BINARY_COL"); + byte[] value = "snowflake_to_the_moon".getBytes(); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + value, type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(byte[].class) + .expectedParsedValue(value) + .expectedSize( + value.length + DEFINITION_LEVEL_ENCODING_BYTE_LEN + BYTE_ARRAY_LENGTH_ENCODING_BYTE_LEN) + .expectedMinMax(value) + .assertMatches(); + } + + @Test + public void parseValueStringToBinary() { + Type type = + Types.primitive(PrimitiveTypeName.BINARY, Repetition.OPTIONAL) + .as(LogicalTypeAnnotation.stringType()) + .named("BINARY_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("BINARY_COL"); + String value = "snowflake_to_the_moon"; + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + value, type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(byte[].class) + .expectedParsedValue(value.getBytes()) + .expectedSize( + value.getBytes().length + + DEFINITION_LEVEL_ENCODING_BYTE_LEN + + BYTE_ARRAY_LENGTH_ENCODING_BYTE_LEN) + .expectedMinMax(value.getBytes()) + .assertMatches(); + } + + @Test + public void parseValueFixed() { + Type type = + Types.primitive(PrimitiveTypeName.FIXED_LEN_BYTE_ARRAY, Repetition.OPTIONAL) + .length(4) + .named("FIXED_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("FIXED_COL"); + byte[] value = "snow".getBytes(); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + value, type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(byte[].class) + .expectedParsedValue(value) + .expectedSize( + 4.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN + BYTE_ARRAY_LENGTH_ENCODING_BYTE_LEN) + .expectedMinMax(value) + .assertMatches(); + } + + @Test + public void parseValueDecimalToFixed() { + Type type = + Types.primitive(PrimitiveTypeName.FIXED_LEN_BYTE_ARRAY, Repetition.OPTIONAL) + .length(9) + .as(LogicalTypeAnnotation.decimalType(10, 20)) + .named("FIXED_COL"); + + RowBufferStats rowBufferStats = new RowBufferStats("FIXED_COL"); + BigDecimal value = new BigDecimal("1234567890.0123456789"); + ParquetBufferValue pv = + IcebergParquetValueParser.parseColumnValueToParquet( + value, type, rowBufferStats, UTC, 0, false); + ParquetValueParserAssertionBuilder.newBuilder() + .parquetBufferValue(pv) + .rowBufferStats(rowBufferStats) + .expectedValueClass(byte[].class) + .expectedParsedValue(value.unscaledValue().toByteArray()) + .expectedSize( + 9.0f + DEFINITION_LEVEL_ENCODING_BYTE_LEN + BYTE_ARRAY_LENGTH_ENCODING_BYTE_LEN) + .expectedMinMax(value.unscaledValue()) + .assertMatches(); + } +} diff --git a/src/test/java/net/snowflake/ingest/streaming/internal/ParquetValueParserAssertionBuilder.java b/src/test/java/net/snowflake/ingest/streaming/internal/ParquetValueParserAssertionBuilder.java new file mode 100644 index 000000000..8480311fa --- /dev/null +++ b/src/test/java/net/snowflake/ingest/streaming/internal/ParquetValueParserAssertionBuilder.java @@ -0,0 +1,104 @@ +/* + * Copyright (c) 2024 Snowflake Computing Inc. All rights reserved. + */ + +package net.snowflake.ingest.streaming.internal; + +import java.math.BigDecimal; +import java.math.BigInteger; +import java.nio.charset.StandardCharsets; +import org.junit.Assert; + +/** Builder that helps to assert parsing of values to parquet types */ +class ParquetValueParserAssertionBuilder { + private ParquetBufferValue parquetBufferValue; + private RowBufferStats rowBufferStats; + private Class valueClass; + private Object value; + private float size; + private Object minMaxStat; + private long currentNullCount; + + static ParquetValueParserAssertionBuilder newBuilder() { + ParquetValueParserAssertionBuilder builder = new ParquetValueParserAssertionBuilder(); + return builder; + } + + ParquetValueParserAssertionBuilder parquetBufferValue(ParquetBufferValue parquetBufferValue) { + this.parquetBufferValue = parquetBufferValue; + return this; + } + + ParquetValueParserAssertionBuilder rowBufferStats(RowBufferStats rowBufferStats) { + this.rowBufferStats = rowBufferStats; + return this; + } + + ParquetValueParserAssertionBuilder expectedValueClass(Class valueClass) { + this.valueClass = valueClass; + return this; + } + + ParquetValueParserAssertionBuilder expectedParsedValue(Object value) { + this.value = value; + return this; + } + + ParquetValueParserAssertionBuilder expectedSize(float size) { + this.size = size; + return this; + } + + public ParquetValueParserAssertionBuilder expectedMinMax(Object minMaxStat) { + this.minMaxStat = minMaxStat; + return this; + } + + public ParquetValueParserAssertionBuilder expectedNullCount(long currentNullCount) { + this.currentNullCount = currentNullCount; + return this; + } + + void assertMatches() { + Assert.assertEquals(valueClass, parquetBufferValue.getValue().getClass()); + if (valueClass.equals(byte[].class)) { + Assert.assertArrayEquals((byte[]) value, (byte[]) parquetBufferValue.getValue()); + } else { + Assert.assertEquals(value, parquetBufferValue.getValue()); + } + Assert.assertEquals(size, parquetBufferValue.getSize(), 0); + if (minMaxStat instanceof BigInteger) { + Assert.assertEquals(minMaxStat, rowBufferStats.getCurrentMinIntValue()); + Assert.assertEquals(minMaxStat, rowBufferStats.getCurrentMaxIntValue()); + return; + } else if (minMaxStat instanceof byte[]) { + Assert.assertArrayEquals((byte[]) minMaxStat, rowBufferStats.getCurrentMinStrValue()); + Assert.assertArrayEquals((byte[]) minMaxStat, rowBufferStats.getCurrentMaxStrValue()); + return; + } else if (valueClass.equals(String.class)) { + // String can have null min/max stats for variant data types + Object min = + rowBufferStats.getCurrentMinStrValue() != null + ? new String(rowBufferStats.getCurrentMinStrValue(), StandardCharsets.UTF_8) + : rowBufferStats.getCurrentMinStrValue(); + Object max = + rowBufferStats.getCurrentMaxStrValue() != null + ? new String(rowBufferStats.getCurrentMaxStrValue(), StandardCharsets.UTF_8) + : rowBufferStats.getCurrentMaxStrValue(); + Assert.assertEquals(minMaxStat, min); + Assert.assertEquals(minMaxStat, max); + return; + } else if (minMaxStat instanceof Double || minMaxStat instanceof BigDecimal) { + Assert.assertEquals(minMaxStat, rowBufferStats.getCurrentMinRealValue()); + Assert.assertEquals(minMaxStat, rowBufferStats.getCurrentMaxRealValue()); + return; + } + throw new IllegalArgumentException( + String.format("Unknown data type for min stat: %s", minMaxStat.getClass())); + } + + void assertNull() { + Assert.assertNull(parquetBufferValue.getValue()); + Assert.assertEquals(currentNullCount, rowBufferStats.getCurrentNullCount()); + } +} diff --git a/src/test/java/net/snowflake/ingest/streaming/internal/SnowflakeParquetValueParserTest.java b/src/test/java/net/snowflake/ingest/streaming/internal/SnowflakeParquetValueParserTest.java index 3aa653960..f3cb82420 100644 --- a/src/test/java/net/snowflake/ingest/streaming/internal/SnowflakeParquetValueParserTest.java +++ b/src/test/java/net/snowflake/ingest/streaming/internal/SnowflakeParquetValueParserTest.java @@ -1,3 +1,7 @@ +/* + * Copyright (c) 2024 Snowflake Computing Inc. All rights reserved. + */ + package net.snowflake.ingest.streaming.internal; import static java.time.ZoneOffset.UTC; @@ -601,98 +605,4 @@ public void parseValueTimeSB16Error() { Assert.assertEquals( "Unknown data type for logical: TIME, physical: SB16.", exception.getMessage()); } - - /** Builder that helps to assert parsing of values to parquet types */ - private static class ParquetValueParserAssertionBuilder { - private ParquetBufferValue parquetBufferValue; - private RowBufferStats rowBufferStats; - private Class valueClass; - private Object value; - private float size; - private Object minMaxStat; - private long currentNullCount; - - static ParquetValueParserAssertionBuilder newBuilder() { - ParquetValueParserAssertionBuilder builder = new ParquetValueParserAssertionBuilder(); - return builder; - } - - ParquetValueParserAssertionBuilder parquetBufferValue(ParquetBufferValue parquetBufferValue) { - this.parquetBufferValue = parquetBufferValue; - return this; - } - - ParquetValueParserAssertionBuilder rowBufferStats(RowBufferStats rowBufferStats) { - this.rowBufferStats = rowBufferStats; - return this; - } - - ParquetValueParserAssertionBuilder expectedValueClass(Class valueClass) { - this.valueClass = valueClass; - return this; - } - - ParquetValueParserAssertionBuilder expectedParsedValue(Object value) { - this.value = value; - return this; - } - - ParquetValueParserAssertionBuilder expectedSize(float size) { - this.size = size; - return this; - } - - public ParquetValueParserAssertionBuilder expectedMinMax(Object minMaxStat) { - this.minMaxStat = minMaxStat; - return this; - } - - public ParquetValueParserAssertionBuilder expectedNullCount(long currentNullCount) { - this.currentNullCount = currentNullCount; - return this; - } - - void assertMatches() { - Assert.assertEquals(valueClass, parquetBufferValue.getValue().getClass()); - if (valueClass.equals(byte[].class)) { - Assert.assertArrayEquals((byte[]) value, (byte[]) parquetBufferValue.getValue()); - } else { - Assert.assertEquals(value, parquetBufferValue.getValue()); - } - Assert.assertEquals(size, parquetBufferValue.getSize(), 0); - if (minMaxStat instanceof BigInteger) { - Assert.assertEquals(minMaxStat, rowBufferStats.getCurrentMinIntValue()); - Assert.assertEquals(minMaxStat, rowBufferStats.getCurrentMaxIntValue()); - return; - } else if (minMaxStat instanceof byte[]) { - Assert.assertArrayEquals((byte[]) minMaxStat, rowBufferStats.getCurrentMinStrValue()); - Assert.assertArrayEquals((byte[]) minMaxStat, rowBufferStats.getCurrentMaxStrValue()); - return; - } else if (valueClass.equals(String.class)) { - // String can have null min/max stats for variant data types - Object min = - rowBufferStats.getCurrentMinStrValue() != null - ? new String(rowBufferStats.getCurrentMinStrValue(), StandardCharsets.UTF_8) - : rowBufferStats.getCurrentMinStrValue(); - Object max = - rowBufferStats.getCurrentMaxStrValue() != null - ? new String(rowBufferStats.getCurrentMaxStrValue(), StandardCharsets.UTF_8) - : rowBufferStats.getCurrentMaxStrValue(); - Assert.assertEquals(minMaxStat, min); - Assert.assertEquals(minMaxStat, max); - return; - } else if (minMaxStat instanceof Double || minMaxStat instanceof BigDecimal) { - Assert.assertEquals(minMaxStat, rowBufferStats.getCurrentMinRealValue()); - Assert.assertEquals(minMaxStat, rowBufferStats.getCurrentMaxRealValue()); - return; - } - throw new IllegalArgumentException( - String.format("Unknown data type for min stat: %s", minMaxStat.getClass())); - } - - void assertNull() { - Assert.assertNull(parquetBufferValue.getValue()); - Assert.assertEquals(currentNullCount, rowBufferStats.getCurrentNullCount()); - } - } }