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

com.alibaba.hologres.client.utils.RecordChecker Maven / Gradle / Ivy

/*
 * Copyright (c) 2022. Alibaba Group Holding Limited
 */

package com.alibaba.hologres.client.utils;

import com.alibaba.hologres.client.exception.ExceptionCode;
import com.alibaba.hologres.client.exception.HoloClientException;
import com.alibaba.hologres.client.model.Column;
import com.alibaba.hologres.client.model.Record;
import com.alibaba.hologres.client.model.TableSchema;

import java.math.BigDecimal;
import java.sql.Time;
import java.sql.Timestamp;
import java.sql.Types;
import java.time.LocalDateTime;
import java.util.Arrays;
import java.util.Date;
import java.util.List;

/**
 * check Input Value is valid.
 * - CONSTRAINT VIOLATION
 */
public class RecordChecker {
	public static void check(Record record) throws HoloClientException {
		TableSchema schema = record.getSchema();
		for (int i = 0; i < schema.getColumnSchema().length; ++i) {
			checkObject(schema, schema.getColumn(i), record.getObject(i));
		}
	}

	private static void throwConstraintViolationException(TableSchema schema, Column column, Object value, String msg, Exception e) throws HoloClientException {
		StringBuilder sb = new StringBuilder();
		sb.append("invalid value [").append(value).append("]");
		if (value != null) {
			sb.append("(").append(value.getClass().getName()).append(")");
		}
		sb.append(" for column ").append(column.getName()).append(" of table ").append(schema.getTableName());
		if (msg != null) {
			sb.append(", reason=").append(msg);
		}
		HoloClientException ret = null;
		if (e != null) {
			ret = new HoloClientException(ExceptionCode.CONSTRAINT_VIOLATION, sb.toString(), e);
		} else {
			ret = new HoloClientException(ExceptionCode.CONSTRAINT_VIOLATION, sb.toString());
		}
		throw ret;

	}

	private static void checkObject(TableSchema schema, Column column, Object value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		} else if (value instanceof Short) {
			checkShort(schema, column, (Short) value);
		} else if (value instanceof Integer) {
			checkInteger(schema, column, (Integer) value);
		} else if (value instanceof Long) {
			checkLong(schema, column, (Long) value);
		} else if (value instanceof BigDecimal) {
			checkBigDecimal(schema, column, (BigDecimal) value);
		} else if (value instanceof Float) {
			checkFloat(schema, column, (Float) value);
		} else if (value instanceof Double) {
			checkDouble(schema, column, (Double) value);
		} else if (value instanceof Boolean) {
			checkBoolean(schema, column, (Boolean) value);
		} else if (value instanceof LocalDateTime) {
			checkLocalDateTime(schema, column, (LocalDateTime) value);
		} else if (value instanceof Timestamp) { // Timestamp must before Date.
			checkTimeStamp(schema, column, (Timestamp) value);
		}  else if (value instanceof Time) { // Time must before Date.
			checkTime(schema, column, (Time) value);
		} else if (value instanceof Date) {
			checkDate(schema, column, (Date) value);
		} else if (value instanceof String[]) {
			checkStringArray(schema, column, (String[]) value);
		} else if (value instanceof int[]) {
			checkIntegerArray(schema, column, (int[]) value);
		} else if (value instanceof Integer[]) {
			checkIntegerArray(schema, column, (Integer[]) value);
		} else if (value instanceof long[]) {
			checkLongArray(schema, column, (long[]) value);
		} else if (value instanceof Long[]) {
			checkLongArray(schema, column, (Long[]) value);
		} else if (value instanceof float[]) {
			checkFloatArray(schema, column, (float[]) value);
		} else if (value instanceof Float[]) {
			checkFloatArray(schema, column, (Float[]) value);
		} else if (value instanceof double[]) {
			checkDoubleArray(schema, column, (double[]) value);
		} else if (value instanceof Double[]) {
			checkDoubleArray(schema, column, (Double[]) value);
		} else if (value instanceof boolean[]) {
			checkBooleanArray(schema, column, (boolean[]) value);
		} else if (value instanceof Boolean[]) {
			checkBooleanArray(schema, column, (Boolean[]) value);
		} else if (value instanceof List) {
			List list = (List) value;
			checkObject(schema, column, list.toArray());
		} else if (value instanceof byte[]) {
			switch (column.getType()) {
				case Types.OTHER:
					if ("roaringbitmap".equals(column.getTypeName())) {
						checkRoaringbitmap(schema, column, (byte[]) value);
						break;
					}
				case Types.BINARY:
					checkBytea(schema, column, (byte[]) value);
				default:
			}
		} else if (value instanceof String) {
			switch (column.getType()) {
				case Types.OTHER:
					if ("json".equals(column.getTypeName())) {
						checkJson(schema, column, (String) value);
						break;
					} else if ("jsonb".equals(column.getTypeName())) {
						checkJsonb(schema, column, (String) value);
						break;
					}
				default:
					checkString(schema, column, value.toString());
			}

		} else {
			checkString(schema, column, value.toString());
		}
	}

	private static void checkNull(TableSchema schema, Column column) throws HoloClientException {
		if (!column.getAllowNull() && column.getDefaultValue() == null) {
			throwConstraintViolationException(schema, column, null, "not allow null value", null);
		}
	}

	private static void checkShort(TableSchema schema, Column column, short value) throws HoloClientException {
		switch (column.getType()) {
			case Types.SMALLINT:
			case Types.INTEGER:
			case Types.BIGINT:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setShort method", null);
		}
	}

	private static void checkInteger(TableSchema schema, Column column, int value) throws HoloClientException {
		switch (column.getType()) {
			case Types.INTEGER:
			case Types.BIGINT:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setInteger method", null);
		}
	}

	private static void checkLong(TableSchema schema, Column column, long value) throws HoloClientException {
		switch (column.getType()) {
			case Types.BIGINT:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setLong method", null);
		}
	}

	private static void checkBigDecimal(TableSchema schema, Column column, BigDecimal value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.NUMERIC:
				if (column.getPrecision() > 0 && column.getPrecision() >= column.getScale()) {
					if (value.precision() - value.scale() > column.getPrecision() - column.getScale()) {
						throwConstraintViolationException(schema, column, value, "A field with precision " + column.getPrecision() + ", scale " + column.getScale() + " must round to an absolute value less than 10^" + (column.getPrecision() - column.getScale()), null);
					}
				}
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setBigDecimal method", null);
		}
	}

	private static void checkBoolean(TableSchema schema, Column column, boolean value) throws HoloClientException {
		switch (column.getType()) {
			case Types.BOOLEAN:
				break;
			case Types.BIT:
				if ("bool".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setBoolean method", null);
		}
	}

	private static void checkFloat(TableSchema schema, Column column, float value) throws HoloClientException {
		switch (column.getType()) {
			case Types.REAL:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setFloat method", null);
		}
	}

	private static void checkDouble(TableSchema schema, Column column, double value) throws HoloClientException {
		switch (column.getType()) {
			case Types.DOUBLE:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setDouble method", null);
		}
	}

	private static void checkDate(TableSchema schema, Column column, Date value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		}
		switch (column.getType()) {
			case Types.DATE:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setDate method", null);
		}
	}

	private static void checkTime(TableSchema schema, Column column, Time value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		}
		switch (column.getType()) {
			case Types.TIME:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setTime method", null);
		}
	}

	private static void checkTimeStamp(TableSchema schema, Column column, Timestamp value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		}
		switch (column.getType()) {
			case Types.TIMESTAMP:
			case Types.TIMESTAMP_WITH_TIMEZONE:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setTimeStamp method", null);
		}
	}

	private static void checkLocalDateTime(TableSchema schema, Column column, LocalDateTime value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		}
		switch (column.getType()) {
			case Types.TIMESTAMP:
			case Types.TIMESTAMP_WITH_TIMEZONE:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setLocalDateTime method", null);
		}
	}

	private static void checkString(TableSchema schema, Column column, String value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		}
		if (value.contains("\u0000")) {
			throwConstraintViolationException(schema, column, value.replaceAll("\u0000", "\\\\u0000"), "invalid byte sequence for encoding \"UTF8\": 0x00", null);
		}
		switch (column.getType()) {
			case Types.VARCHAR:
			case Types.CHAR:
				if (column.getPrecision() > 0) {
					int precision = column.getPrecision();
					if (value.length() > precision) {
						throwConstraintViolationException(schema, column, value, "value too long for type character varying(" + precision + ")", null);
					}
				}
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setString method", null);
		}
	}

	private static void checkJson(TableSchema schema, Column column, String value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		}
		if (value.contains("\u0000")) {
			throwConstraintViolationException(schema, column, value.replaceAll("\u0000", "\\\\u0000"), "invalid byte sequence for encoding \"UTF8\": 0x00", null);
		}
		if (value.contains("\\u0000")) {
			throwConstraintViolationException(schema, column, value, "invalid byte sequence for encoding \"UTF8\": 0x00", null);
		}
		switch (column.getType()) {
			case Types.OTHER:
				if ("json".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setJson method", null);
		}
	}

	private static void checkJsonb(TableSchema schema, Column column, String value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		}
		if (value.contains("\u0000")) {
			throwConstraintViolationException(schema, column, value.replaceAll("\u0000", "\\\\u0000"), "invalid byte sequence for encoding \"UTF8\": 0x00", null);
		}
		if (value.contains("\\u0000")) {
			throwConstraintViolationException(schema, column, value, "invalid byte sequence for encoding \"UTF8\": 0x00", null);
		}
		switch (column.getType()) {
			case Types.OTHER:
				if ("jsonb".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setJsonb method", null);
		}
	}

	private static void checkBytea(TableSchema schema, Column column, byte[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		}

		switch (column.getType()) {
			case Types.BINARY:
				break;
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setBytea method", null);
		}
	}

	private static void checkRoaringbitmap(TableSchema schema, Column column, byte[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
		}

		switch (column.getType()) {
			case Types.OTHER:
				if ("roaringbitmap".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setRoaringbitmap method", null);
		}
	}

	private static void checkStringArray(TableSchema schema, Column column, String[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		for (String str : value) {
			if (str == null) {
				throwConstraintViolationException(schema, column, Arrays.toString(value), "Not support null value in array", null);
			} else if (str.contains("\u0000")) {
				throwConstraintViolationException(schema, column, Arrays.toString(value).replaceAll("\u0000", "\\\\u0000"), "invalid byte sequence for encoding \"UTF8\": 0x00", null);
			}
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_text".equals(column.getTypeName())) {
					break;
				} else if ("_varchar".equals(column.getTypeName())) {
					if (column.getPrecision() > 0) {
						int precision = column.getPrecision();
						for (String str : value) {
							if (str.length() > precision) {
								throwConstraintViolationException(schema, column, Arrays.toString(value), "value too long for type character varying(" + precision + ")", null);
							}
						}
					}
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setStringArray method", null);
		}
	}

	private static void checkBooleanArray(TableSchema schema, Column column, boolean[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_bool".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setBooleanArray method", null);
		}
	}

	private static void checkBooleanArray(TableSchema schema, Column column, Boolean[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_bool".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setBooleanArray method", null);
		}
	}

	private static void checkFloatArray(TableSchema schema, Column column, float[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_float4".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setFloatArray method", null);
		}
	}

	private static void checkFloatArray(TableSchema schema, Column column, Float[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_float4".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setFloatArray method", null);
		}
	}

	private static void checkDoubleArray(TableSchema schema, Column column, double[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_float8".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setDoubleArray method", null);
		}
	}

	private static void checkDoubleArray(TableSchema schema, Column column, Double[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_float8".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setDoubleArray method", null);
		}
	}

	private static void checkIntegerArray(TableSchema schema, Column column, int[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_int4".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setIntegerArray method", null);
		}
	}

	private static void checkIntegerArray(TableSchema schema, Column column, Integer[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_int4".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setIntegerArray method", null);
		}
	}

	private static void checkLongArray(TableSchema schema, Column column, long[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_int8".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setLongArray method", null);
		}
	}

	private static void checkLongArray(TableSchema schema, Column column, Long[] value) throws HoloClientException {
		if (value == null) {
			checkNull(schema, column);
			return;
		}
		switch (column.getType()) {
			case Types.ARRAY:
				if ("_int8".equals(column.getTypeName())) {
					break;
				}
			default:
				throwConstraintViolationException(schema, column, value, "unsupported type " + column.getTypeName() + " for setLongArray method", null);
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy