Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support spatial extension types #168

Merged
merged 1 commit into from
Mar 25, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -560,6 +560,7 @@ add_library(duckdb_java SHARED
src/jni/duckdb_java.cpp
src/jni/functions.cpp
src/jni/refs.cpp
src/jni/types.cpp
src/jni/util.cpp
${DUCKDB_SRC_FILES})

Expand Down
44 changes: 3 additions & 41 deletions src/jni/duckdb_java.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@
#include "duckdb/parser/parsed_data/create_type_info.hpp"
#include "functions.hpp"
#include "refs.hpp"
#include "types.hpp"
#include "util.hpp"

#include <cstdint>
Expand Down Expand Up @@ -396,42 +397,6 @@ void _duckdb_jdbc_free_result(JNIEnv *env, jclass, jobject res_ref_buf) {
}
}

static std::string type_to_jduckdb_type(LogicalType logical_type) {
switch (logical_type.id()) {
case LogicalTypeId::DECIMAL: {

uint8_t width = 0;
uint8_t scale = 0;
logical_type.GetDecimalProperties(width, scale);
std::string width_scale = std::to_string(width) + std::string(";") + std::to_string(scale);

auto physical_type = logical_type.InternalType();
switch (physical_type) {
case PhysicalType::INT16: {
string res = std::string("DECIMAL16;") + width_scale;
return res;
}
case PhysicalType::INT32: {
string res = std::string("DECIMAL32;") + width_scale;
return res;
}
case PhysicalType::INT64: {
string res = std::string("DECIMAL64;") + width_scale;
return res;
}
case PhysicalType::INT128: {
string res = std::string("DECIMAL128;") + width_scale;
return res;
}
default:
return std::string("no physical type found");
}
} break;
default:
return logical_type.ToString();
}
}

static jobject build_meta(JNIEnv *env, size_t column_count, size_t n_param, const duckdb::vector<string> &names,
const duckdb::vector<LogicalType> &types, StatementProperties properties) {
auto name_array = env->NewObjectArray(column_count, J_String, nullptr);
Expand Down Expand Up @@ -515,6 +480,7 @@ jobjectArray _duckdb_jdbc_fetch(JNIEnv *env, jclass, jobject res_ref_buf, jobjec

return vec_array;
}

jobject ProcessVector(JNIEnv *env, Connection *conn_ref, Vector &vec, idx_t row_count) {
auto type_str = env->NewStringUTF(type_to_jduckdb_type(vec.GetType()).c_str());
// construct nullmask
Expand All @@ -530,11 +496,7 @@ jobject ProcessVector(JNIEnv *env, Connection *conn_ref, Vector &vec, idx_t row_
jobject constlen_data = nullptr;
jobjectArray varlen_data = nullptr;

// this allows us to treat aliased (usually extension) types as strings
auto type = vec.GetType();
auto type_id = type.HasAlias() ? LogicalTypeId::UNKNOWN : type.id();

switch (type_id) {
switch (vec.GetType().id()) {
case LogicalTypeId::BOOLEAN:
constlen_data = env->NewDirectByteBuffer(FlatVector::GetData(vec), row_count * sizeof(bool));
break;
Expand Down
45 changes: 45 additions & 0 deletions src/jni/types.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
#include "types.hpp"

#include <string>
#include <vector>

std::string type_to_jduckdb_type(duckdb::LogicalType logical_type) {
switch (logical_type.id()) {
case duckdb::LogicalTypeId::DECIMAL: {

uint8_t width = 0;
uint8_t scale = 0;
logical_type.GetDecimalProperties(width, scale);
std::string width_scale = std::to_string(width) + std::string(";") + std::to_string(scale);

auto physical_type = logical_type.InternalType();
switch (physical_type) {
case duckdb::PhysicalType::INT16: {
std::string res = std::string("DECIMAL16;") + width_scale;
return res;
}
case duckdb::PhysicalType::INT32: {
std::string res = std::string("DECIMAL32;") + width_scale;
return res;
}
case duckdb::PhysicalType::INT64: {
std::string res = std::string("DECIMAL64;") + width_scale;
return res;
}
case duckdb::PhysicalType::INT128: {
std::string res = std::string("DECIMAL128;") + width_scale;
return res;
}
default:
return std::string("no physical type found");
}
} break;
default:
// JSON requires special handling because it is mapped
// to JsonNode class
if (logical_type.IsJSONType()) {
return logical_type.GetAlias();
}
return duckdb::EnumUtil::ToString(logical_type.id());
}
}
5 changes: 5 additions & 0 deletions src/jni/types.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
#pragma once

#include "duckdb.hpp"

std::string type_to_jduckdb_type(duckdb::LogicalType logical_type);
39 changes: 13 additions & 26 deletions src/test/java/org/duckdb/TestDuckDBJDBC.java
Original file line number Diff line number Diff line change
Expand Up @@ -11,14 +11,7 @@
import static org.duckdb.DuckDBDriver.DUCKDB_USER_AGENT_PROPERTY;
import static org.duckdb.DuckDBDriver.JDBC_STREAM_RESULTS;
import static org.duckdb.DuckDBTimestamp.localDateTimeFromTimestamp;
import static org.duckdb.test.Assertions.assertEquals;
import static org.duckdb.test.Assertions.assertFalse;
import static org.duckdb.test.Assertions.assertNotNull;
import static org.duckdb.test.Assertions.assertNull;
import static org.duckdb.test.Assertions.assertThrows;
import static org.duckdb.test.Assertions.assertThrowsMaybe;
import static org.duckdb.test.Assertions.assertTrue;
import static org.duckdb.test.Assertions.fail;
import static org.duckdb.test.Assertions.*;
import static org.duckdb.test.Runner.runTests;

import java.math.BigDecimal;
Expand Down Expand Up @@ -1302,7 +1295,7 @@ public static void test_lots_of_decimals() throws Exception {
conn.close();
}

public static void test_big_data() throws Exception {
public static void test_lots_of_big_data() throws Exception {
Connection conn = DriverManager.getConnection(JDBC_URL);
Statement stmt = conn.createStatement();
int rows = 10000;
Expand Down Expand Up @@ -4187,21 +4180,6 @@ private static Map<String, Object> structToMap(DuckDBStruct actual) throws SQLEx
return result;
}

private static <T> void assertListsEqual(List<T> actual, List<T> expected) throws Exception {
assertListsEqual(actual, expected, "");
}

private static <T> void assertListsEqual(List<T> actual, List<T> expected, String label) throws Exception {
assertEquals(actual.size(), expected.size());

ListIterator<T> itera = actual.listIterator();
ListIterator<T> itere = expected.listIterator();

while (itera.hasNext()) {
assertEquals(itera.next(), itere.next(), label);
}
}

public static void test_cancel() throws Exception {
ExecutorService service = Executors.newFixedThreadPool(1);
try (Connection conn = DriverManager.getConnection(JDBC_URL); Statement stmt = conn.createStatement()) {
Expand Down Expand Up @@ -4300,7 +4278,7 @@ public static void test_invalid_execute_calls() throws Exception {
}
}

public static void test_race() throws Exception {
public static void test_lots_of_races() throws Exception {
try (Connection connection = DriverManager.getConnection(JDBC_URL)) {
ExecutorService executorService = Executors.newFixedThreadPool(10);

Expand Down Expand Up @@ -4799,6 +4777,15 @@ public static void test_typed_connection_properties() throws Exception {
}

public static void main(String[] args) throws Exception {
System.exit(runTests(args, TestDuckDBJDBC.class, TestExtensionTypes.class));
String arg1 = args.length > 0 ? args[0] : "";
final int statusCode;
if (arg1.startsWith("Test")) {
Class<?> clazz = Class.forName("org.duckdb." + arg1);
statusCode = runTests(new String[0], clazz);
} else {
// extension installation fails on CI, Spatial test is temporary disabled
statusCode = runTests(args, TestDuckDBJDBC.class, TestExtensionTypes.class /*, TestSpatial.class */);
}
System.exit(statusCode);
}
}
17 changes: 9 additions & 8 deletions src/test/java/org/duckdb/TestExtensionTypes.java
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,7 @@
import static org.duckdb.TestDuckDBJDBC.JDBC_URL;
import static org.duckdb.test.Assertions.assertEquals;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.sql.Statement;
import java.sql.Types;
import java.sql.*;

public class TestExtensionTypes {

Expand All @@ -21,8 +16,14 @@ public static void test_extension_type() throws Exception {
try (ResultSet rs = stmt.executeQuery(
"SELECT {\"hello\": 'foo', \"world\": 'bar'}::test_type, '\\xAA'::byte_test_type")) {
rs.next();
assertEquals(rs.getObject(1), "{'hello': foo, 'world': bar}");
assertEquals(rs.getObject(2), "\\xAA");
Struct struct = (Struct) rs.getObject(1);
Object[] attrs = struct.getAttributes();
assertEquals(attrs[0], "foo");
assertEquals(attrs[1], "bar");
Blob blob = rs.getBlob(2);
byte[] bytes = blob.getBytes(1, (int) blob.length());
assertEquals(bytes.length, 1);
assertEquals(bytes[0] & 0xff, 0xaa);
}
}
}
Expand Down
Loading
Loading