Skip to content

Commit

Permalink
Merge branch 'master' into tzhang-si-fixes
Browse files Browse the repository at this point in the history
  • Loading branch information
sfc-gh-tzhang authored Sep 22, 2023
2 parents efa8e4e + 687327b commit aea5519
Show file tree
Hide file tree
Showing 6 changed files with 101 additions and 3 deletions.
2 changes: 1 addition & 1 deletion pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
<!-- Arifact name and version information -->
<groupId>net.snowflake</groupId>
<artifactId>snowflake-ingest-sdk</artifactId>
<version>2.0.3</version>
<version>2.0.4-SNAPSHOT</version>
<packaging>jar</packaging>
<name>Snowflake Ingest SDK</name>
<description>Snowflake Ingest SDK</description>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -110,7 +110,7 @@ public class RequestBuilder {
// Don't change!
public static final String CLIENT_NAME = "SnowpipeJavaSDK";

public static final String DEFAULT_VERSION = "2.0.3";
public static final String DEFAULT_VERSION = "2.0.4-SNAPSHOT";

public static final String JAVA_USER_AGENT = "JAVA";

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
import java.util.Map;
import java.util.concurrent.CompletableFuture;
import javax.annotation.Nullable;
import net.snowflake.ingest.streaming.internal.ColumnProperties;

/**
* A logical partition that represents a connection to a single Snowflake table, data will be
Expand All @@ -17,7 +18,7 @@
* at the same time. When a new channel is opened, all previously opened channels with the same name
* are invalidated (this applies for the table globally. not just in a single JVM). In order to
* ingest data from multiple threads/clients/applications, we recommend opening multiple channels,
* each with a different name. There is no limit on the number of channels that can be opened.
* each with a different name.
*
* <p>Thread safety note: Implementations of this interface are required to be thread safe.
*/
Expand Down Expand Up @@ -253,4 +254,13 @@ InsertValidationResponse insertRows(
*/
@Nullable
String getLatestCommittedOffsetToken();

/**
* Gets the table schema associated with this channel. Note that this is the table schema at the
* time of a channel open event. The schema may be changed on the Snowflake side in which case
* this will continue to show an old schema version until the channel is re-opened.
*
* @return map representing Column Name -> Column Properties
*/
Map<String, ColumnProperties> getTableSchema();
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
package net.snowflake.ingest.streaming.internal;

/**
* Class that encapsulates column properties. These are the same properties showed in the output of
* <a href="https://docs.snowflake.com/en/sql-reference/sql/show-columns">SHOW COLUMNS</a>. Note
* that this is slightly different than the internal column metadata used elsewhere in this SDK.
*/
public class ColumnProperties {
private String type;

private String logicalType;

private Integer precision;

private Integer scale;

private Integer byteLength;

private Integer length;

private boolean nullable;

ColumnProperties(ColumnMetadata columnMetadata) {
this.type = columnMetadata.getType();
this.logicalType = columnMetadata.getLogicalType();
this.precision = columnMetadata.getPrecision();
this.scale = columnMetadata.getScale();
this.byteLength = columnMetadata.getByteLength();
this.length = columnMetadata.getLength();
this.nullable = columnMetadata.getNullable();
}

public String getType() {
return type;
}

public String getLogicalType() {
return logicalType;
}

public Integer getPrecision() {
return precision;
}

public Integer getScale() {
return scale;
}

public Integer getByteLength() {
return byteLength;
}

public Integer getLength() {
return length;
}

public boolean isNullable() {
return nullable;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
import java.time.ZoneId;
import java.time.ZoneOffset;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.LinkedList;
import java.util.List;
Expand Down Expand Up @@ -51,6 +52,9 @@ class SnowflakeStreamingIngestChannelInternal<T> implements SnowflakeStreamingIn
// State of the channel that will be shared with its underlying buffer
private final ChannelRuntimeState channelState;

// Internal map of column name -> column properties
private final Map<String, ColumnProperties> tableColumns;

/**
* Constructor for TESTING ONLY which allows us to set the test mode
*
Expand Down Expand Up @@ -122,6 +126,7 @@ class SnowflakeStreamingIngestChannelInternal<T> implements SnowflakeStreamingIn
this::collectRowSize,
channelState,
new ClientBufferParameters(owningClient));
this.tableColumns = new HashMap<>();
logger.logInfo(
"Channel={} created for table={}",
this.channelFlushContext.getName(),
Expand Down Expand Up @@ -298,6 +303,7 @@ public CompletableFuture<Void> close() {
void setupSchema(List<ColumnMetadata> columns) {
logger.logDebug("Setup schema for channel={}, schema={}", getFullyQualifiedName(), columns);
this.rowBuffer.setupSchema(columns);
columns.forEach(c -> tableColumns.putIfAbsent(c.getName(), new ColumnProperties(c)));
}

/**
Expand Down Expand Up @@ -391,6 +397,12 @@ public String getLatestCommittedOffsetToken() {
return response.getPersistedOffsetToken();
}

/** Returns a map of column name -> datatype for the table the channel is bound to */
@Override
public Map<String, ColumnProperties> getTableSchema() {
return this.tableColumns;
}

/** Check whether we need to throttle the insertRows API */
void throttleInsertIfNeeded(MemoryInfoProvider memoryInfoProvider) {
int retry = 0;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -495,6 +495,22 @@ public void testOpenChannelSuccessResponse() throws Exception {
Assert.assertEquals(dbName, channel.getDBName());
Assert.assertEquals(schemaName, channel.getSchemaName());
Assert.assertEquals(tableName, channel.getTableName());

Assert.assertTrue(channel.getTableSchema().containsKey("C1"));
Assert.assertEquals("NUMBER(38,0)", channel.getTableSchema().get("C1").getType());
Assert.assertEquals("FIXED", channel.getTableSchema().get("C1").getLogicalType());
Assert.assertEquals(38, (int) channel.getTableSchema().get("C1").getPrecision());
Assert.assertEquals(0, (int) channel.getTableSchema().get("C1").getScale());
Assert.assertNull(channel.getTableSchema().get("C1").getByteLength());
Assert.assertTrue(channel.getTableSchema().get("C1").isNullable());

Assert.assertTrue(channel.getTableSchema().containsKey("C2"));
Assert.assertEquals("NUMBER(38,0)", channel.getTableSchema().get("C2").getType());
Assert.assertEquals("FIXED", channel.getTableSchema().get("C2").getLogicalType());
Assert.assertEquals(38, (int) channel.getTableSchema().get("C2").getPrecision());
Assert.assertEquals(0, (int) channel.getTableSchema().get("C2").getScale());
Assert.assertNull(channel.getTableSchema().get("C2").getByteLength());
Assert.assertTrue(channel.getTableSchema().get("C2").isNullable());
}

@Test
Expand Down

0 comments on commit aea5519

Please sign in to comment.