
org.dinky.shaded.paimon.data.GenericRow Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.dinky.shaded.paimon.data;
import org.dinky.shaded.paimon.annotation.Public;
import org.dinky.shaded.paimon.types.RowKind;
import org.dinky.shaded.paimon.types.RowType;
import java.io.Serializable;
import java.util.Arrays;
import java.util.Objects;
import static org.dinky.shaded.paimon.utils.Preconditions.checkNotNull;
/**
* An internal data structure representing data of {@link RowType}.
*
* {@link GenericRow} is a generic implementation of {@link InternalRow} which is backed by an
* array of Java {@link Object}. A {@link GenericRow} can have an arbitrary number of fields of
* different types. The fields in a row can be accessed by position (0-based) using either the
* generic {@link #getField(int)} or type-specific getters (such as {@link #getInt(int)}). A field
* can be updated by the generic {@link #setField(int, Object)}.
*
*
Note: All fields of this data structure must be internal data structures. See {@link
* InternalRow} for more information about internal data structures.
*
*
The fields in {@link GenericRow} can be null for representing nullability.
*
* @since 0.4.0
*/
@Public
public final class GenericRow implements InternalRow, Serializable {
private static final long serialVersionUID = 1L;
/** The array to store the actual internal format values. */
private final Object[] fields;
/** The kind of change that a row describes in a changelog. */
private RowKind kind;
/**
* Creates an instance of {@link GenericRow} with given kind and number of fields.
*
*
Initially, all fields are set to null.
*
*
Note: All fields of the row must be internal data structures.
*
* @param kind kind of change that this row describes in a changelog
* @param arity number of fields
*/
public GenericRow(RowKind kind, int arity) {
this.fields = new Object[arity];
this.kind = kind;
}
/**
* Creates an instance of {@link GenericRow} with given number of fields.
*
*
Initially, all fields are set to null. By default, the row describes a {@link
* RowKind#INSERT} in a changelog.
*
*
Note: All fields of the row must be internal data structures.
*
* @param arity number of fields
*/
public GenericRow(int arity) {
this.fields = new Object[arity];
this.kind = RowKind.INSERT; // INSERT as default
}
/**
* Sets the field value at the given position.
*
*
Note: The given field value must be an internal data structures. Otherwise the {@link
* GenericRow} is corrupted and may throw exception when processing. See {@link InternalRow} for
* more information about internal data structures.
*
*
The field value can be null for representing nullability.
*/
public void setField(int pos, Object value) {
this.fields[pos] = value;
}
/**
* Returns the field value at the given position.
*
*
Note: The returned value is in internal data structure. See {@link InternalRow} for more
* information about internal data structures.
*
*
The returned field value can be null for representing nullability.
*/
public Object getField(int pos) {
return this.fields[pos];
}
@Override
public int getFieldCount() {
return fields.length;
}
@Override
public RowKind getRowKind() {
return kind;
}
@Override
public void setRowKind(RowKind kind) {
checkNotNull(kind);
this.kind = kind;
}
@Override
public boolean isNullAt(int pos) {
return this.fields[pos] == null;
}
@Override
public boolean getBoolean(int pos) {
return (boolean) this.fields[pos];
}
@Override
public byte getByte(int pos) {
return (byte) this.fields[pos];
}
@Override
public short getShort(int pos) {
return (short) this.fields[pos];
}
@Override
public int getInt(int pos) {
return (int) this.fields[pos];
}
@Override
public long getLong(int pos) {
return (long) this.fields[pos];
}
@Override
public float getFloat(int pos) {
return (float) this.fields[pos];
}
@Override
public double getDouble(int pos) {
return (double) this.fields[pos];
}
@Override
public BinaryString getString(int pos) {
return (BinaryString) this.fields[pos];
}
@Override
public Decimal getDecimal(int pos, int precision, int scale) {
return (Decimal) this.fields[pos];
}
@Override
public Timestamp getTimestamp(int pos, int precision) {
return (Timestamp) this.fields[pos];
}
@Override
public byte[] getBinary(int pos) {
return (byte[]) this.fields[pos];
}
@Override
public InternalArray getArray(int pos) {
return (InternalArray) this.fields[pos];
}
@Override
public InternalMap getMap(int pos) {
return (InternalMap) this.fields[pos];
}
@Override
public InternalRow getRow(int pos, int numFields) {
return (InternalRow) this.fields[pos];
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (!(o instanceof GenericRow)) {
return false;
}
GenericRow that = (GenericRow) o;
return kind == that.kind && Arrays.deepEquals(fields, that.fields);
}
@Override
public int hashCode() {
int result = Objects.hash(kind);
result = 31 * result + Arrays.deepHashCode(fields);
return result;
}
@Override
public String toString() {
StringBuilder sb = new StringBuilder();
sb.append(kind.shortString()).append("(");
for (int i = 0; i < fields.length; i++) {
if (i != 0) {
sb.append(",");
}
sb.append(arrayAwareToString(fields[i]));
}
sb.append(")");
return sb.toString();
}
private static String arrayAwareToString(Object o) {
final String arrayString = Arrays.deepToString(new Object[] {o});
return arrayString.substring(1, arrayString.length() - 1);
}
// ----------------------------------------------------------------------------------------
// Utilities
// ----------------------------------------------------------------------------------------
/**
* Creates an instance of {@link GenericRow} with given field values.
*
*
By default, the row describes a {@link RowKind#INSERT} in a changelog.
*
*
Note: All fields of the row must be internal data structures.
*/
public static GenericRow of(Object... values) {
GenericRow row = new GenericRow(values.length);
for (int i = 0; i < values.length; ++i) {
row.setField(i, values[i]);
}
return row;
}
/**
* Creates an instance of {@link GenericRow} with given kind and field values.
*
*
Note: All fields of the row must be internal data structures.
*/
public static GenericRow ofKind(RowKind kind, Object... values) {
GenericRow row = new GenericRow(kind, values.length);
for (int i = 0; i < values.length; ++i) {
row.setField(i, values[i]);
}
return row;
}
}