All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.postgresql.jdbc.PgArray Maven / Gradle / Ivy

There is a newer version: 42.7.3-yb-2
Show newest version
/*
 * Copyright (c) 2004, PostgreSQL Global Development Group
 * See the LICENSE file in the project root for more information.
 */

package org.postgresql.jdbc;

import static org.postgresql.util.internal.Nullness.castNonNull;

import org.postgresql.core.BaseConnection;
import org.postgresql.core.BaseStatement;
import org.postgresql.core.Field;
import org.postgresql.core.Oid;
import org.postgresql.core.Tuple;
import org.postgresql.jdbc.ArrayDecoding.PgArrayList;
import org.postgresql.jdbc2.ArrayAssistantRegistry;
import org.postgresql.util.ByteConverter;
import org.postgresql.util.GT;
import org.postgresql.util.PSQLException;
import org.postgresql.util.PSQLState;

import org.checkerframework.checker.nullness.qual.Nullable;

import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;

/**
 * 

Array is used collect one column of query result data.

* *

Read a field of type Array into either a natively-typed Java array object or a ResultSet. * Accessor methods provide the ability to capture array slices.

* *

Other than the constructor all methods are direct implementations of those specified for * java.sql.Array. Please refer to the javadoc for java.sql.Array for detailed descriptions of the * functionality and parameters of the methods of this class.

* * @see ResultSet#getArray */ public class PgArray implements java.sql.Array { static { ArrayAssistantRegistry.register(Oid.UUID, new UUIDArrayAssistant()); ArrayAssistantRegistry.register(Oid.UUID_ARRAY, new UUIDArrayAssistant()); } /** * A database connection. */ protected @Nullable BaseConnection connection; /** * The OID of this field. */ private final int oid; /** * Field value as String. */ protected @Nullable String fieldString; /** * Value of field as {@link PgArrayList}. Will be initialized only once within * {@link #buildArrayList(String)}. */ protected ArrayDecoding.@Nullable PgArrayList arrayList; protected byte @Nullable [] fieldBytes; private PgArray(BaseConnection connection, int oid) throws SQLException { this.connection = connection; this.oid = oid; } /** * Create a new Array. * * @param connection a database connection * @param oid the oid of the array datatype * @param fieldString the array data in string form * @throws SQLException if something wrong happens */ public PgArray(BaseConnection connection, int oid, @Nullable String fieldString) throws SQLException { this(connection, oid); this.fieldString = fieldString; } /** * Create a new Array. * * @param connection a database connection * @param oid the oid of the array datatype * @param fieldBytes the array data in byte form * @throws SQLException if something wrong happens */ public PgArray(BaseConnection connection, int oid, byte @Nullable [] fieldBytes) throws SQLException { this(connection, oid); this.fieldBytes = fieldBytes; } private BaseConnection getConnection() { return castNonNull(connection); } @SuppressWarnings("return.type.incompatible") public Object getArray() throws SQLException { return getArrayImpl(1, 0, null); } @SuppressWarnings("return.type.incompatible") public Object getArray(long index, int count) throws SQLException { return getArrayImpl(index, count, null); } @SuppressWarnings("return.type.incompatible") public Object getArrayImpl(Map> map) throws SQLException { return getArrayImpl(1, 0, map); } @SuppressWarnings("return.type.incompatible") public Object getArray(Map> map) throws SQLException { return getArrayImpl(map); } @SuppressWarnings("return.type.incompatible") public Object getArray(long index, int count, @Nullable Map> map) throws SQLException { return getArrayImpl(index, count, map); } public @Nullable Object getArrayImpl(long index, int count, @Nullable Map> map) throws SQLException { // for now maps aren't supported. if (map != null && !map.isEmpty()) { throw org.postgresql.Driver.notImplemented(this.getClass(), "getArrayImpl(long,int,Map)"); } // array index is out of range if (index < 1) { throw new PSQLException(GT.tr("The array index is out of range: {0}", index), PSQLState.DATA_ERROR); } if (fieldBytes != null) { return readBinaryArray(fieldBytes, (int) index, count); } if (fieldString == null) { return null; } final PgArrayList arrayList = buildArrayList(fieldString); if (count == 0) { count = arrayList.size(); } // array index out of range if ((index - 1) + count > arrayList.size()) { throw new PSQLException( GT.tr("The array index is out of range: {0}, number of elements: {1}.", index + count, (long) arrayList.size()), PSQLState.DATA_ERROR); } return buildArray(arrayList, (int) index, count); } private Object readBinaryArray(byte[] fieldBytes, int index, int count) throws SQLException { return ArrayDecoding.readBinaryArray(index, count, fieldBytes, getConnection()); } private ResultSet readBinaryResultSet(byte[] fieldBytes, int index, int count) throws SQLException { int dimensions = ByteConverter.int4(fieldBytes, 0); // int flags = ByteConverter.int4(fieldBytes, 4); // bit 0: 0=no-nulls, 1=has-nulls int elementOid = ByteConverter.int4(fieldBytes, 8); int pos = 12; int[] dims = new int[dimensions]; for (int d = 0; d < dimensions; ++d) { dims[d] = ByteConverter.int4(fieldBytes, pos); pos += 4; /* int lbound = ByteConverter.int4(fieldBytes, pos); */ pos += 4; } if (count > 0 && dimensions > 0) { dims[0] = Math.min(count, dims[0]); } List rows = new ArrayList(); Field[] fields = new Field[2]; storeValues(fieldBytes, rows, fields, elementOid, dims, pos, 0, index); BaseStatement stat = (BaseStatement) getConnection() .createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); return stat.createDriverResultSet(fields, rows); } private int storeValues(byte[] fieldBytes, List rows, Field[] fields, int elementOid, final int[] dims, int pos, final int thisDimension, int index) throws SQLException { // handle an empty array if (dims.length == 0) { fields[0] = new Field("INDEX", Oid.INT4); fields[0].setFormat(Field.BINARY_FORMAT); fields[1] = new Field("VALUE", elementOid); fields[1].setFormat(Field.BINARY_FORMAT); for (int i = 1; i < index; ++i) { int len = ByteConverter.int4(fieldBytes, pos); pos += 4; if (len != -1) { pos += len; } } } else if (thisDimension == dims.length - 1) { fields[0] = new Field("INDEX", Oid.INT4); fields[0].setFormat(Field.BINARY_FORMAT); fields[1] = new Field("VALUE", elementOid); fields[1].setFormat(Field.BINARY_FORMAT); for (int i = 1; i < index; ++i) { int len = ByteConverter.int4(fieldBytes, pos); pos += 4; if (len != -1) { pos += len; } } for (int i = 0; i < dims[thisDimension]; ++i) { byte[][] rowData = new byte[2][]; rowData[0] = new byte[4]; ByteConverter.int4(rowData[0], 0, i + index); rows.add(new Tuple(rowData)); int len = ByteConverter.int4(fieldBytes, pos); pos += 4; if (len == -1) { continue; } rowData[1] = new byte[len]; System.arraycopy(fieldBytes, pos, rowData[1], 0, rowData[1].length); pos += len; } } else { fields[0] = new Field("INDEX", Oid.INT4); fields[0].setFormat(Field.BINARY_FORMAT); fields[1] = new Field("VALUE", oid); fields[1].setFormat(Field.BINARY_FORMAT); int nextDimension = thisDimension + 1; int dimensionsLeft = dims.length - nextDimension; for (int i = 1; i < index; ++i) { pos = calcRemainingDataLength(fieldBytes, dims, pos, elementOid, nextDimension); } for (int i = 0; i < dims[thisDimension]; ++i) { byte[][] rowData = new byte[2][]; rowData[0] = new byte[4]; ByteConverter.int4(rowData[0], 0, i + index); rows.add(new Tuple(rowData)); int dataEndPos = calcRemainingDataLength(fieldBytes, dims, pos, elementOid, nextDimension); int dataLength = dataEndPos - pos; rowData[1] = new byte[12 + 8 * dimensionsLeft + dataLength]; ByteConverter.int4(rowData[1], 0, dimensionsLeft); System.arraycopy(fieldBytes, 4, rowData[1], 4, 8); System.arraycopy(fieldBytes, 12 + nextDimension * 8, rowData[1], 12, dimensionsLeft * 8); System.arraycopy(fieldBytes, pos, rowData[1], 12 + dimensionsLeft * 8, dataLength); pos = dataEndPos; } } return pos; } private int calcRemainingDataLength(byte[] fieldBytes, int[] dims, int pos, int elementOid, int thisDimension) { if (thisDimension == dims.length - 1) { for (int i = 0; i < dims[thisDimension]; ++i) { int len = ByteConverter.int4(fieldBytes, pos); pos += 4; if (len == -1) { continue; } pos += len; } } else { pos = calcRemainingDataLength(fieldBytes, dims, elementOid, pos, thisDimension + 1); } return pos; } /** * Build {@link ArrayList} from field's string input. As a result of this method * {@link #arrayList} is build. Method can be called many times in order to make sure that array * list is ready to use, however {@link #arrayList} will be set only once during first call. */ private synchronized PgArrayList buildArrayList(String fieldString) throws SQLException { if (arrayList == null) { arrayList = ArrayDecoding.buildArrayList(fieldString, getConnection().getTypeInfo().getArrayDelimiter(oid)); } return arrayList; } /** * Convert {@link ArrayList} to array. * * @param input list to be converted into array */ private Object buildArray(ArrayDecoding.PgArrayList input, int index, int count) throws SQLException { final BaseConnection connection = getConnection(); return ArrayDecoding.readStringArray(index, count, connection.getTypeInfo().getPGArrayElement(oid), input, connection); } public int getBaseType() throws SQLException { return getConnection().getTypeInfo().getSQLType(getBaseTypeName()); } public String getBaseTypeName() throws SQLException { int elementOID = getConnection().getTypeInfo().getPGArrayElement(oid); return castNonNull(getConnection().getTypeInfo().getPGType(elementOID)); } public java.sql.ResultSet getResultSet() throws SQLException { return getResultSetImpl(1, 0, null); } public java.sql.ResultSet getResultSet(long index, int count) throws SQLException { return getResultSetImpl(index, count, null); } public ResultSet getResultSet(@Nullable Map> map) throws SQLException { return getResultSetImpl(map); } public ResultSet getResultSet(long index, int count, @Nullable Map> map) throws SQLException { return getResultSetImpl(index, count, map); } public ResultSet getResultSetImpl(@Nullable Map> map) throws SQLException { return getResultSetImpl(1, 0, map); } public ResultSet getResultSetImpl(long index, int count, @Nullable Map> map) throws SQLException { // for now maps aren't supported. if (map != null && !map.isEmpty()) { throw org.postgresql.Driver.notImplemented(this.getClass(), "getResultSetImpl(long,int,Map)"); } // array index is out of range if (index < 1) { throw new PSQLException(GT.tr("The array index is out of range: {0}", index), PSQLState.DATA_ERROR); } if (fieldBytes != null) { return readBinaryResultSet(fieldBytes, (int) index, count); } final PgArrayList arrayList = buildArrayList(castNonNull(fieldString)); if (count == 0) { count = arrayList.size(); } // array index out of range if ((--index) + count > arrayList.size()) { throw new PSQLException( GT.tr("The array index is out of range: {0}, number of elements: {1}.", index + count, (long) arrayList.size()), PSQLState.DATA_ERROR); } List rows = new ArrayList(); Field[] fields = new Field[2]; // one dimensional array if (arrayList.dimensionsCount <= 1) { // array element type final int baseOid = getConnection().getTypeInfo().getPGArrayElement(oid); fields[0] = new Field("INDEX", Oid.INT4); fields[1] = new Field("VALUE", baseOid); for (int i = 0; i < count; i++) { int offset = (int) index + i; byte[] @Nullable [] t = new byte[2][0]; String v = (String) arrayList.get(offset); t[0] = getConnection().encodeString(Integer.toString(offset + 1)); t[1] = v == null ? null : getConnection().encodeString(v); rows.add(new Tuple(t)); } } else { // when multi-dimensional fields[0] = new Field("INDEX", Oid.INT4); fields[1] = new Field("VALUE", oid); for (int i = 0; i < count; i++) { int offset = (int) index + i; byte[] @Nullable [] t = new byte[2][0]; Object v = arrayList.get(offset); t[0] = getConnection().encodeString(Integer.toString(offset + 1)); t[1] = v == null ? null : getConnection().encodeString(toString((ArrayDecoding.PgArrayList) v)); rows.add(new Tuple(t)); } } BaseStatement stat = (BaseStatement) getConnection() .createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); return stat.createDriverResultSet(fields, rows); } @SuppressWarnings("nullness") public @Nullable String toString() { if (fieldString == null && fieldBytes != null) { try { Object array = readBinaryArray(fieldBytes, 1, 0); final ArrayEncoding.ArrayEncoder arraySupport = ArrayEncoding.getArrayEncoder(array); assert arraySupport != null; fieldString = arraySupport.toArrayString(connection.getTypeInfo().getArrayDelimiter(oid), array); } catch (SQLException e) { fieldString = "NULL"; // punt } } return fieldString; } /** * Convert array list to PG String representation (e.g. {0,1,2}). */ private String toString(ArrayDecoding.PgArrayList list) throws SQLException { if (list == null) { return "NULL"; } StringBuilder b = new StringBuilder().append('{'); char delim = getConnection().getTypeInfo().getArrayDelimiter(oid); for (int i = 0; i < list.size(); i++) { Object v = list.get(i); if (i > 0) { b.append(delim); } if (v == null) { b.append("NULL"); } else if (v instanceof ArrayDecoding.PgArrayList) { b.append(toString((ArrayDecoding.PgArrayList) v)); } else { escapeArrayElement(b, (String) v); } } b.append('}'); return b.toString(); } public static void escapeArrayElement(StringBuilder b, String s) { b.append('"'); for (int j = 0; j < s.length(); j++) { char c = s.charAt(j); if (c == '"' || c == '\\') { b.append('\\'); } b.append(c); } b.append('"'); } public boolean isBinary() { return fieldBytes != null; } public byte @Nullable [] toBytes() { return fieldBytes; } public void free() throws SQLException { connection = null; fieldString = null; fieldBytes = null; arrayList = null; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy