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

com.alibaba.hologres.client.impl.util.ConnectionUtil Maven / Gradle / Ivy

There is a newer version: 2.5.5
Show newest version
/*
 * Copyright (c) 2021. Alibaba Group Holding Limited
 */

package com.alibaba.hologres.client.impl.util;

import com.alibaba.hologres.client.model.Column;
import com.alibaba.hologres.client.model.HoloVersion;
import com.alibaba.hologres.client.model.Partition;
import com.alibaba.hologres.client.model.TableName;
import com.alibaba.hologres.client.model.TableSchema;
import com.alibaba.hologres.client.utils.IdentifierUtil;
import org.postgresql.PGProperty;
import org.postgresql.jdbc.PgConnection;
import org.postgresql.util.PSQLState;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * connection工具类.
 */
public class ConnectionUtil {

	public static final Logger LOGGER = LoggerFactory.getLogger(ConnectionUtil.class);
	static Pattern holoVersionPattern = Pattern.compile("release-([^ )]*)");
	static Pattern hgVersionPattern = Pattern.compile("Hologres ([^ -]*)");
    private static String optionProperty = "options=";
	private static String fixedOption = "type=fixed%20";

	public static void refreshMeta(Connection conn, int timeout) throws SQLException {
		try (Statement stat = conn.createStatement()) {
			stat.execute("select hologres.hg_internal_refresh_meta(" + timeout + ")");
		}
	}

	/**
	 * CheckMeta的结果.
	 */
	public static class CheckMetaResult {
		boolean updated; //是否最新
		String msg; //不是最新的话,原因是什么

		public CheckMetaResult(boolean updated, String msg) {
			this.updated = updated;
			this.msg = msg;
		}

		public boolean isUpdated() {
			return updated;
		}

		public String getMsg() {
			return msg;
		}
	}

	//目前还有问题,哪个版本修复不确定,拍个2.0.0先,后面版本确定了再改
	private static final HoloVersion CHECK_TABLE_META_SUPPORTED_MIN_VERSION = new HoloVersion(1, 1, 50);

	public static CheckMetaResult checkMeta(Connection conn, HoloVersion version, String fullName, int timeout) throws SQLException {
		if (version.compareTo(CHECK_TABLE_META_SUPPORTED_MIN_VERSION) >= 0) {

			try (Statement stat = conn.createStatement()) {
				stat.setQueryTimeout(timeout);
				try (ResultSet rs = stat.executeQuery("select hologres.hg_internal_check_table_meta('" + fullName + "')")) {
					if (rs.next()) {
						String msg = rs.getString(1);
						boolean ok = "Check meta succeeded".equals(msg);
						if (ok) {
							return new CheckMetaResult(ok, msg);
						} else {
							refreshMeta(conn, timeout);
							return new CheckMetaResult(true, null);
						}
					} else {
						return new CheckMetaResult(false, "hologres.hg_internal_check_table_meta return 0 rows");
					}
				} catch (SQLException e) {
					if (PSQLState.UNDEFINED_TABLE.getState().equals(e.getSQLState())) {
						refreshMeta(conn, timeout);
						return new CheckMetaResult(true, null);
					} else {
						throw e;
					}
				}
			} catch (SQLException e) {
				if (PSQLState.QUERY_CANCELED.getState().equals(e.getSQLState())) {
					return new CheckMetaResult(false, "table is lock by other query which request a AccessExclusiveLock");
				} else if (e.getMessage().contains("Failed to get table") && e.getMessage().contains("from StoreMaster")) {
					return new CheckMetaResult(false, "Failed to get table from StoreMaster, maybe still in replay after a truncate");
				} else {
					throw e;
				}
			}
		} else {
			refreshMeta(conn, timeout);
			return new CheckMetaResult(true, null);
		}
	}

	public static HoloVersion getHoloVersion(Connection conn) throws SQLException {
		try (Statement stmt = conn.createStatement()) {
			try {
				String hgverStr = parseSingleCell(stmt.executeQuery("select hg_version()"));
				return parseHgVersion(hgverStr);
			} catch (Exception e) {
				//ignore error as hg_version() does NOT exist in earlier hologres instances.
			}
		}
		try (Statement stmt = conn.createStatement()) {
			String verStr = parseSingleCell(stmt.executeQuery("select version()"));
			return parseHoloVersion(verStr);
		}
	}

	public static HoloVersion parseHgVersion(String verStr) throws Exception {
		//Hologres 0.11.1 (tag: release-0.11.x build: Release,Skylake,clang,no-coroutine)
		Matcher matcher = hgVersionPattern.matcher(verStr);
		if (matcher.find()) {
			String v = matcher.group(1);
			HoloVersion hv = new HoloVersion(v);
			if (!hv.isUndefined()) {
				return hv;
			}
		}
		throw new Exception("Failed to parse hg_version() result");
	}

	public static HoloVersion parseHoloVersion(String verStr) {
		//"PostgreSQL 11.3 (Release-build@4bf700d062 on origin/release-0.8.x) on x86_64-pc-linux-gnu, compiled by x86_64-pc-linux-gnu-gcc (GCC) 8.3.0, 64-bit"
		Matcher matcher = holoVersionPattern.matcher(verStr);
		if (matcher.find()) {
			String v = matcher.group(1);
			return new HoloVersion(v);
		}
		return null;
	}

	public static String getDatabase(Connection conn) throws SQLException {
		try (Statement stat = conn.createStatement()) {
			return parseSingleCell(stat.executeQuery("select current_database()"));
		}
	}

	// SELECT n.nspname as Schema, c.relname as Name,
	// part.partstrat,
	// part.partnatts,
	// part.partattrs
	// FROM pg_catalog.pg_class c
	// JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace
	// JOIN pg_catalog.pg_partitioned_table part ON c.oid = part.partrelid
	// where n.nspname='public' and c.relname='test_message'
	// limit 1;
	//  schema |     name     | partstrat | partnatts | partattrs
	// --------+--------------+-----------+-----------+-----------
	//  public | test_message | l         |         1 | 1

	// SELECT attname
	// FROM pg_attribute
	// WHERE attrelid = 'public.test_message'::regclass
	// AND attnum = 1;
	// attname
	// ---------
	// ds
	public static String getPartitionColumnName(Connection conn, TableName tableName) throws SQLException {
		StringBuilder sb = new StringBuilder(512);
		sb.append("SELECT n.nspname as Schema, c.relname as Name,\n");
		sb.append("    part.partstrat,\n");
		sb.append("    part.partnatts,\n");
		sb.append("    part.partattrs\n");
		sb.append("FROM pg_catalog.pg_class c\n");
		sb.append("JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace\n");
		sb.append("JOIN pg_catalog.pg_partitioned_table part ON c.oid = part.partrelid\n");
		sb.append("where n.nspname=? and c.relname=? \n");
		sb.append("limit 1;\n");
		String sql = sb.toString();

		int partColumnPos = -1;
		try (PreparedStatement stmt = conn.prepareStatement(sql)) {
			stmt.setString(1, tableName.getSchemaName());
			stmt.setString(2, tableName.getTableName());
			ResultSet rs = stmt.executeQuery();

			if (rs.next()) {
				String strategyStr = rs.getString("partstrat");
				if (!"l".equals(strategyStr)) {

					throw new SQLException("Only LIST partition is supported in holo.");
				}

				String partColumnStr = rs.getString("partattrs");
				partColumnPos = Integer.parseInt(partColumnStr);
			}
		}

		String partColumnName = null;
		sql = "SELECT attname FROM pg_attribute WHERE attrelid =?::regclass AND attnum = ?;";
		if (partColumnPos > 0) {
			try (PreparedStatement stmt = conn.prepareStatement(sql)) {
				stmt.setString(1, tableName.getFullName());
				stmt.setInt(2, partColumnPos);
				ResultSet rs = stmt.executeQuery();

				if (rs.next()) {
					return rs.getString("attname");
				}
			}
		}
		return partColumnName;
	}

	// Query all child tables
	// with inh as (
	//     SELECT i.inhrelid, i.inhparent
	//     FROM pg_catalog.pg_class c
	//     LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace
	//     LEFT JOIN pg_catalog.pg_inherits i on c.oid=i.inhparent
	//     where n.nspname='public' and c.relname='test_message'
	// )
	// select
	//     n.nspname as schema_name,
	//     c.relname as table_name,
	//     inh.inhrelid, inh.inhparent, p.partstrat,
	//     pg_get_expr(c.relpartbound, c.oid, true) as part_expr,
	//     p.partdefid,
	//     p.partnatts,
	//     p.partattrs
	// from inh
	// join pg_catalog.pg_class c on inh.inhrelid = c.oid
	// join pg_catalog.pg_namespace n on c.relnamespace = n.oid
	// join pg_partitioned_table p on p.partrelid = inh.inhparent;
	//
	//  schema_name |    table_name    | inhrelid | inhparent | partstrat |       part_expr       | partdefid | partnatts | partattrs
	// -------------+------------------+----------+-----------+-----------+-----------------------+-----------+-----------+-----------
	//  public      | test_message_foo |    65655 |     65650 | l         | FOR VALUES IN ('foo') |         0 |         1 | 1
	//  public      | test_message_bar |    65663 |     65650 | l         | FOR VALUES IN ('bar') |         0 |         1 | 1
	// (2 rows)
	public static Partition getPartition(Connection conn, String schemaName, String tableName, String partValue, boolean isStr) throws SQLException {
		StringBuilder sb = new StringBuilder(512);
		sb.append("with inh as ( \n");
		sb.append("    SELECT i.inhrelid, i.inhparent \n");
		sb.append("    FROM pg_catalog.pg_class c \n");
		sb.append("    LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace \n");
		sb.append("    LEFT JOIN pg_catalog.pg_inherits i on c.oid=i.inhparent \n");
		sb.append("    where n.nspname=? and c.relname=? \n");
		sb.append(") \n");
		sb.append("select \n");
		sb.append("    n.nspname as schema_name, \n");
		sb.append("    c.relname as table_name, \n");
		sb.append("    inh.inhrelid, inh.inhparent, p.partstrat, \n");
		sb.append("    pg_get_expr(c.relpartbound, c.oid, true) as part_expr, \n");
		sb.append("    p.partdefid, \n");
		sb.append("    p.partnatts, \n");
		sb.append("    p.partattrs \n");
		sb.append("from inh \n");
		sb.append("join pg_catalog.pg_class c on inh.inhrelid = c.oid \n");
		sb.append("join pg_catalog.pg_namespace n on c.relnamespace = n.oid \n");
		sb.append("join pg_partitioned_table p on p.partrelid = inh.inhparent where pg_get_expr(c.relpartbound, c.oid, true)=? limit 1 \n");

		String sql = sb.toString();
		ResultSet rs = null;
		Partition partition = null;
		try (PreparedStatement stmt = conn.prepareStatement(sql)) {

			stmt.setString(1, schemaName);
			stmt.setString(2, tableName);
			stmt.setString(3, "FOR VALUES IN (" + (isStr ? "'" : "") + partValue + (isStr ? "'" : "") + ")");
			rs = stmt.executeQuery();
			if (rs.next()) {

				String strategyStr = rs.getString("partstrat");
				if (!"l".equals(strategyStr)) {

					throw new SQLException("Only LIST partition is supported in holo.");
				}
				partition = new Partition();
				partition.setParentSchemaName(schemaName);
				partition.setParentTableName(tableName);
				String schema = rs.getString("schema_name");
				String table = rs.getString("table_name");
				partition.setSchemaName(schema);
				partition.setTableName(table);
				partition.setPartitionValue(partValue);

			}

			return partition;
		}

	}

	public static Partition retryCreatePartitionChildTable(Connection conn, String schemaName, String tableName, String partValue, boolean isStr) throws SQLException {

		int retry = 0;
		while (true) {
			Statement stmt = null;
			String childSchemaName = schemaName;
			String childTableName = retry
					== 0 ? String.format("%s_%s", tableName, partValue) : String.format("%s_%s_%d", tableName, partValue, System.currentTimeMillis());

			try {
				String valueStr = null;
				if (isStr) {
					valueStr = String.format("'%s'", partValue);
				} else {
					valueStr = partValue;
				}
				String sql = String.format("create table %s.%s partition of %s.%s for values in (%s);",
						IdentifierUtil.quoteIdentifier(childSchemaName, true), IdentifierUtil.quoteIdentifier(childTableName, true),
						IdentifierUtil.quoteIdentifier(schemaName, true), IdentifierUtil.quoteIdentifier(tableName, true),
						valueStr);

				stmt = conn.createStatement();
				stmt.execute(sql);

				Partition partition = new Partition();
				partition.setTableName(childTableName);
				partition.setSchemaName(childSchemaName);
				partition.setParentTableName(tableName);
				partition.setParentSchemaName(schemaName);
				partition.setPartitionValue(valueStr);
				return partition;
			} catch (SQLException e) {
				String alreadyExistMsg = String.format("relation \"%s\" already exists", childTableName);
				if (e.getMessage().indexOf(alreadyExistMsg) != -1 && retry < 20) {
					try {
						Thread.sleep(3000);
					} catch (InterruptedException ex) {
					}
					// retry
					++retry;
					continue;
				}
				throw e;
			} finally {
				if (stmt != null) {
					stmt.close();
				}
			}

		}
	}

	public static TableSchema getTableSchema(Connection conn, TableName tableName) throws SQLException {
		String[] columns = null;
		int[] types = null;
		String[] typeNames = null;
		DatabaseMetaData metaData = conn.getMetaData();
		List primaryKeyList = new ArrayList<>();
		try (ResultSet rs = metaData.getPrimaryKeys(null, tableName.getSchemaName(), tableName.getTableName())) {
			while (rs.next()) {
				primaryKeyList.add(rs.getString(4));
			}
		}
		List columnList = new ArrayList<>();
		String escape = metaData.getSearchStringEscape();
		// getColumns方法的后三个参数都是LIKE表达式使用的pattern,因此需要对这里传入的schemaName和tableName中的特殊字符进行转义。
		// https://www.postgresql.org/docs/current/functions-matching.html#FUNCTIONS-LIKE PG文档中明确LIKE的特殊字符只有%以及_(还有转义字符本身)。
		try (ResultSet rs = metaData.getColumns(null, escapePattern(tableName.getSchemaName(), escape), escapePattern(tableName.getTableName(), escape), "%")) {
			while (rs.next()) {
				Column column = new Column();
				column.setName(rs.getString(4));
				column.setType(rs.getInt(5));
				column.setTypeName(rs.getString(6));
				column.setPrecision(rs.getInt(7));
				column.setScale(rs.getInt(9));
				column.setAllowNull(rs.getInt(11) == 1);
				column.setComment(rs.getString(12));
				column.setDefaultValue(rs.getObject(13));
				column.setArrayType(column.getTypeName().startsWith("_"));
				column.setPrimaryKey(primaryKeyList.contains(column.getName()));
				columnList.add(column);
			}
		}

		String partitionColumnName = getPartitionColumnName(conn, tableName);

		String sql =
				"select property_key,property_value from hologres.hg_table_properties where table_namespace=? and table_name=? and property_key in "
						+ "('distribution_key','table_id','schema_version','orientation','clustering_key','segment_key','bitmap_columns','dictionary_encoding_columns','time_to_live_in_seconds','binlog.level')";
		String[] distributionKeys = null;
		String tableId = null;
		String schemaVersion = null;
		Map properties = new HashMap<>();
		try (PreparedStatement stat = conn.prepareStatement(sql)) {
			stat.setString(1, tableName.getSchemaName());
			stat.setString(2, tableName.getTableName());

			try (ResultSet rs = stat.executeQuery()) {
				while (rs.next()) {
					String propertyName = rs.getString(1);
					String propertyValue = rs.getString(2);
					properties.put(propertyName, propertyValue);
				}
			}
			tableId = properties.get("table_id");
			schemaVersion = properties.get("schema_version");
		}
		if (properties.size() == 0) {
			throw new SQLException("can not found table " + tableName.getFullName());
		}
		if (tableId == null) {
			throw new SQLException("table " + tableName.getFullName() + " has no table_id");
		}
		if (schemaVersion == null) {
			throw new SQLException("table " + tableName.getFullName() + " has no schemaVersion");
		}
		TableSchema.Builder builder = new TableSchema.Builder(tableId, schemaVersion);

		builder.setPartitionColumnName(partitionColumnName);
		builder.setColumns(columnList);
		builder.setTableName(tableName);
		builder.setNotExist(false);
		builder.setSensitive(true);
		for (Map.Entry entry : properties.entrySet()) {
			String key = entry.getKey();
			String value = entry.getValue();
			switch (key) {
				case "distribution_key":
					if ("".equals(value)) {
						throw new SQLException("empty distribution_key is not supported.");
					}
					builder.setDistributionKeys(value.split(","));
					break;
				case "orientation":
					builder.setOrientation(value);
					break;
				case "clustering_key":
					builder.setClusteringKey(value.split(","));
					break;
				case "segment_key":
					builder.setSegmentKey(value.split(","));
					break;
				case "bitmap_columns":
					builder.setBitmapIndexKey(value.split(","));
					break;
				case "dictionary_encoding_columns":
					builder.setDictionaryEncoding(value.split(","));
					break;
				case "time_to_live_in_seconds":
					builder.setLifecycle(Long.parseLong(value));
					break;
				case "binlog.level":
					builder.setBinlogLevel(value);
					break;
				default:
			}
		}
		TableSchema tableSchema = builder.build();
		tableSchema.calculateProperties();
		return tableSchema;
	}

	private static String parseSingleCell(ResultSet rs) throws SQLException {
		String ret = null;
		if (rs.next()) {
			ret = rs.getString(1);
		}
		return ret;
	}

	private static String escapePattern(String pattern, String escape) {
		return pattern.replace(escape,  escape + escape)
				.replace("%", escape + "%")
				.replace("_", escape + "_");
	}

	/**
	 * 返回直连fe的jdbc url,如果网络连通可以不走vip.
	 *
	 * @return 直连fe的jdbc url
	 * @throws SQLException
	 */
	public static String getDirectConnectionJdbcUrl(String originalJdbcUrl, Properties info) throws SQLException {
		LOGGER.info("Try to connect {} for getting fe endpoint", originalJdbcUrl);
		String endpoint = "";

		// 由于消费binlog的info中可能设置REPLICATION参数,无法支持select,因此直接使用username和password创建连接
		String username = info.getProperty(PGProperty.USER.getName());
		String password = info.getProperty(PGProperty.PASSWORD.getName());
		try (PgConnection conn = DriverManager.getConnection(originalJdbcUrl, username, password).unwrap(PgConnection.class)) {
			String sql = "select inet_server_addr(), inet_server_port()";
			try (Statement stat = conn.createStatement()) {
				try (ResultSet rs = stat.executeQuery(sql)) {
					while (rs.next()) {
						endpoint = rs.getString(1) + ":" + rs.getString(2);
					}
				}
			}
		}
		String directConnectionJdbcUrl = ConnectionUtil.replaceJdbcUrlEndpoint(originalJdbcUrl, endpoint);
		LOGGER.info("Get the direct connection jdbc url {}", directConnectionJdbcUrl);
		return directConnectionJdbcUrl;
	}

	public static String replaceJdbcUrlEndpoint(String originalUrl, String newEndpoint) {
		String replacement = "//" + newEndpoint + "/";
		return originalUrl.replaceFirst("//\\S+/", replacement);
	}

	public static String generateFixedUrl(String url) {
		StringBuilder sb = new StringBuilder(url);
		int index = sb.lastIndexOf(optionProperty);
		if (index > -1) {
			// In fact fixed fe will ignore url parameters, but we guarantee the correctness of the url here.
			sb.insert(index + optionProperty.length(), fixedOption);
		} else {
			sb.append(url.contains("?") ? "&" : "?").append(optionProperty).append(fixedOption);
		}
		return sb.toString();
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy