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

com.hundsun.lightdb.unisql.proxy.jdbc.UnisqlConnection Maven / Gradle / Ivy

There is a newer version: 24.1.7.0-beta-2
Show newest version
package com.hundsun.lightdb.unisql.proxy.jdbc;

import com.hundsun.lightdb.unisql.golang.Transformer;

import com.hundsun.lightdb.unisql.golang.VariableParameter;
import com.hundsun.lightdb.unisql.model.UnisqlProperties;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
import lombok.EqualsAndHashCode;
import lombok.Getter;
import lombok.ToString;
import lombok.extern.slf4j.Slf4j;

import java.sql.Array;
import java.sql.Blob;
import java.sql.CallableStatement;
import java.sql.Clob;
import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.NClob;
import java.sql.PreparedStatement;
import java.sql.SQLClientInfoException;
import java.sql.SQLException;
import java.sql.SQLWarning;
import java.sql.SQLXML;
import java.sql.Savepoint;
import java.sql.Statement;
import java.sql.Struct;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.Executor;

/**
 * @author zhuxd
 * @date 2023/1/30
 */
@Getter
@ToString
@EqualsAndHashCode
@Slf4j
@SuppressFBWarnings({"SQL_INJECTION_JDBC", "EXTERNAL_CONFIG_CONTROL"})
public class UnisqlConnection implements Connection {

    public static final String ORACLE_JDBC_DRIVER_ORACLE_CONNECTION = "oracle.jdbc.driver.OracleConnection";
    private final DbType sourceDialect;
    private final DbType targetDialect;
    private final String dataBase;

    @Override
    public Statement createStatement() throws SQLException {
        return new UnisqlStatement(this, delegate.createStatement());
    }

    @Override
    public PreparedStatement prepareStatement(String sql) throws SQLException {
        return new UnisqlPreparedStatement(this, delegate.prepareStatement(transformSql(sql, true)), sql);
    }

    public String transformSql(String sql) {
        return transformSql(sql, false);
    }

    private String transformSql(String sql, boolean isPrecompiled) {
        long start = System.nanoTime();

        String transform;
        // http发送sql到`比对中心`服务
        if (Transformer.IS_COMPARE) {
            // 跳过`统一sql运行时sql转换`步骤,原因:内部客户uf30是多套测试环境,不需要在运行时候进行sql转换(由一种sql方言转换为另一种方言)
            if (UnisqlProperties.skipUnisqlRuntimeConvertProperty) {
                return sql;
            }
            // 比对服务开启时,预编译的SQL会调用两次;第一次预编译sql时,此参数为1 不调用比对服务,第二次获取到绑定变量值后,此参数为0 调用比对服务。
            VariableParameter variableParameter = VariableParameter.builder().dbName(dataBase).build();
            if (isPrecompiled) {
                variableParameter.setIsPrecompiled(1);
            }
            transform = Transformer.parse(sql, sourceDialect.name(), targetDialect.name(), variableParameter);
        } else {
            transform = Transformer.parse(sql, sourceDialect.name(), targetDialect.name());
        }

        long cost = System.nanoTime() - start;
        if (log.isDebugEnabled()) {
            log.debug("[UNISQL][transformSql]cost:{} ns. Source db is {}, Target db is {}, From:\n{}\nTo:\n{}. ", cost,
                    sourceDialect.name(), targetDialect.name(), sql, transform);
        }
        return transform;
    }

    @Override
    public CallableStatement prepareCall(String sql) throws SQLException {
        throw new SQLException("not implemented");
        //		return delegate.prepareCall(sql);
    }

    @Override
    public String nativeSQL(String sql) throws SQLException {
        return delegate.nativeSQL(sql);
    }

    @Override
    public void setAutoCommit(boolean autoCommit) throws SQLException {
        delegate.setAutoCommit(autoCommit);
    }

    @Override
    public boolean getAutoCommit() throws SQLException {
        return delegate.getAutoCommit();
    }

    @Override
    public void commit() throws SQLException {
        delegate.commit();
    }

    @Override
    public void rollback() throws SQLException {
        delegate.rollback();
    }

    @Override
    public void close() throws SQLException {
        delegate.close();
    }

    @Override
    public boolean isClosed() throws SQLException {
        return delegate.isClosed();
    }

    @Override
    public DatabaseMetaData getMetaData() throws SQLException {
        return delegate.getMetaData();
    }

    @Override
    public void setReadOnly(boolean readOnly) throws SQLException {
        delegate.setReadOnly(readOnly);
    }

    @Override
    public boolean isReadOnly() throws SQLException {
        return delegate.isReadOnly();
    }

    @Override
    public void setCatalog(String catalog) throws SQLException {
        delegate.setCatalog(catalog);
    }

    @Override
    public String getCatalog() throws SQLException {
        return delegate.getCatalog();
    }

    @Override
    public void setTransactionIsolation(int level) throws SQLException {
        delegate.setTransactionIsolation(level);
    }

    @Override
    public int getTransactionIsolation() throws SQLException {
        return delegate.getTransactionIsolation();
    }

    @Override
    public SQLWarning getWarnings() throws SQLException {
        return delegate.getWarnings();
    }

    @Override
    public void clearWarnings() throws SQLException {
        delegate.clearWarnings();
    }

    @Override
    public Statement createStatement(int resultSetType, int resultSetConcurrency) throws SQLException {
        return new UnisqlStatement(this, delegate.createStatement(resultSetType, resultSetConcurrency));
    }

    @Override
    public PreparedStatement prepareStatement(String sql, int resultSetType, int resultSetConcurrency) throws SQLException {
        return new UnisqlPreparedStatement(this,
                delegate.prepareStatement(transformSql(sql, true), resultSetType, resultSetConcurrency), sql);
    }

    @Override
    public CallableStatement prepareCall(String sql, int resultSetType, int resultSetConcurrency) throws SQLException {
        throw new SQLException("not implemented");
        //		return delegate.prepareCall(sql, resultSetType, resultSetConcurrency);
    }

    @Override
    public Map> getTypeMap() throws SQLException {
        return delegate.getTypeMap();
    }

    @Override
    public void setTypeMap(Map> map) throws SQLException {
        delegate.setTypeMap(map);
    }

    @Override
    public void setHoldability(int holdability) throws SQLException {
        delegate.setHoldability(holdability);
    }

    @Override
    public int getHoldability() throws SQLException {
        return delegate.getHoldability();
    }

    @Override
    public Savepoint setSavepoint() throws SQLException {
        return delegate.setSavepoint();
    }

    @Override
    public Savepoint setSavepoint(String name) throws SQLException {
        return delegate.setSavepoint(name);
    }

    @Override
    public void rollback(Savepoint savepoint) throws SQLException {
        delegate.rollback(savepoint);
    }

    @Override
    public void releaseSavepoint(Savepoint savepoint) throws SQLException {
        delegate.releaseSavepoint(savepoint);
    }

    @Override
    public Statement createStatement(int resultSetType, int resultSetConcurrency, int resultSetHoldability) throws SQLException {
        return new UnisqlStatement(this, delegate.createStatement(resultSetType, resultSetConcurrency,
                resultSetHoldability));
    }

    @Override
    public PreparedStatement prepareStatement(String sql, int resultSetType, int resultSetConcurrency,
                                              int resultSetHoldability) throws SQLException {
        return new UnisqlPreparedStatement(this, delegate.prepareStatement(transformSql(sql, true), resultSetType,
                resultSetConcurrency, resultSetHoldability), sql);
    }

    @Override
    public CallableStatement prepareCall(String sql, int resultSetType, int resultSetConcurrency,
                                         int resultSetHoldability) throws SQLException {
        throw new SQLException("not implemented");
        //		return delegate.prepareCall(sql, resultSetType, resultSetConcurrency, resultSetHoldability);
    }

    @Override
    public PreparedStatement prepareStatement(String sql, int autoGeneratedKeys) throws SQLException {
        return new UnisqlPreparedStatement(this, delegate.prepareStatement(transformSql(sql, true), autoGeneratedKeys), sql);
    }

    @Override
    public PreparedStatement prepareStatement(String sql, int[] columnIndexes) throws SQLException {
        return new UnisqlPreparedStatement(this, delegate.prepareStatement(transformSql(sql, true), columnIndexes), sql);
    }

    @Override
    public PreparedStatement prepareStatement(String sql, String[] columnNames) throws SQLException {
        return new UnisqlPreparedStatement(this, delegate.prepareStatement(transformSql(sql, true), columnNames), sql);
    }

    @Override
    public Clob createClob() throws SQLException {
        return delegate.createClob();
    }

    @Override
    public Blob createBlob() throws SQLException {
        return delegate.createBlob();
    }

    @Override
    public NClob createNClob() throws SQLException {
        return delegate.createNClob();
    }

    @Override
    public SQLXML createSQLXML() throws SQLException {
        return delegate.createSQLXML();
    }

    @Override
    public boolean isValid(int timeout) throws SQLException {
        return delegate.isValid(timeout);
    }

    @Override
    public void setClientInfo(String name, String value) throws SQLClientInfoException {
        delegate.setClientInfo(name, value);
    }

    @Override
    public void setClientInfo(Properties properties) throws SQLClientInfoException {
        delegate.setClientInfo(properties);
    }

    @Override
    public String getClientInfo(String name) throws SQLException {
        return delegate.getClientInfo(name);
    }

    @Override
    public Properties getClientInfo() throws SQLException {
        return delegate.getClientInfo();
    }

    @Override
    public Array createArrayOf(String typeName, Object[] elements) throws SQLException {
        return delegate.createArrayOf(typeName, elements);
    }

    @Override
    public Struct createStruct(String typeName, Object[] attributes) throws SQLException {
        return delegate.createStruct(typeName, attributes);
    }

    @Override
    public void setSchema(String schema) throws SQLException {
        delegate.setSchema(schema);
    }

    @Override
    public String getSchema() throws SQLException {
        return delegate.getSchema();
    }

    @Override
    public void abort(Executor executor) throws SQLException {
        delegate.abort(executor);
    }

    @Override
    public void setNetworkTimeout(Executor executor, int milliseconds) throws SQLException {
        delegate.setNetworkTimeout(executor, milliseconds);
    }

    @Override
    public int getNetworkTimeout() throws SQLException {
        return delegate.getNetworkTimeout();
    }

    @Override
    public  T unwrap(Class iface) throws SQLException {
        return delegate.unwrap(iface);
    }

    @Override
    public boolean isWrapperFor(Class iface) throws SQLException {

        /**
         * 1 兼容研发中心本地缓存组件(localcache)版本(1.1.32.1)代码中方法java.sql.Wrapper.isWrapperFor入参类型`oracle.jdbc.driver.OracleConnection.class`(这是个抽象类)
         * 2 delegate.isWrapperFor(`oracle.jdbc.driver.OracleConnection.class`)会抛出异常`java.sql.SQLException: Object does not wrap anything with requested interface(对象不使用请求的接口包装任何内容)
         * 	at oracle.jdbc.driver.OracleConnection.isWrapperFor(OracleConnection.java:260) ~[ojdbc8-19.3.0.0.jar!/:19.3.0.0.0]`
         * @Author: liangdong30629
         * @Date: 2024/7/10 10:27
         */
        if (ORACLE_JDBC_DRIVER_ORACLE_CONNECTION.equals(iface.getName())) {
            // 参考数据源hikari判断逻辑:com.zaxxer.hikari.pool.ProxyConnection.isWrapperFor
            return iface.isInstance(delegate);
        }
        return delegate.isWrapperFor(iface);
    }

    private final Connection delegate;

    public UnisqlConnection(Connection delegate, DbType sourceDialect, DbType targetDialect, String dataBase) {
        this.delegate = delegate;
        this.sourceDialect = sourceDialect;
        this.targetDialect = targetDialect;
        this.dataBase = dataBase;
    }

    public Connection getDelegate() {
        return delegate;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy