com.caucho.quercus.mysql.MysqlConnectionImpl Maven / Gradle / Ivy
/*
* Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
*
* This file is part of Resin(R) Open Source
*
* Each copy or derived work must preserve the copyright notice and this
* notice unmodified.
*
* Resin Open Source is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Resin Open Source is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
* of NON-INFRINGEMENT. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with Resin Open Source; if not, write to the
*
* Free Software Foundation, Inc.
* 59 Temple Place, Suite 330
* Boston, MA 02111-1307 USA
*
* @author Scott Ferguson
*/
package com.caucho.quercus.mysql;
import com.caucho.quercus.lib.db.QuercusConnection;
import com.caucho.util.*;
import com.caucho.vfs.*;
import java.io.*;
import java.net.*;
import java.sql.*;
import java.util.*;
import java.util.concurrent.Executor;
import java.util.logging.*;
import javax.sql.*;
/**
* Special Quercus Mysql connection.
*/
public class MysqlConnectionImpl implements QuercusConnection {
private static final Logger log
= Logger.getLogger(MysqlConnectionImpl.class.getName());
private static final L10N L = new L10N(MysqlConnectionImpl.class);
private static final int CLIENT_LONG_PASSWORD = 1;
private static final int CLIENT_FOUND_ROWS = 2;
private static final int CLIENT_LONG_FLAG = 4;
private static final int CLIENT_CONNECT_WITH_DB = 8;
private static final int CLIENT_COMPRESS = 32;
private static final int CLIENT_PROTOCOL_41 = 512;
private static final int CLIENT_SSL = 2048;
private static final int CLIENT_SECURE_CONNECTION = 32768;
private static final int CLIENT_MULTI_STATEMENTS = 65536;
private static final int CLIENT_MULTI_RESULTS = 131072;
private static final int UTF8_MB3 = 33;
private static final int COM_QUERY = 0x03;
private QuercusMysqlDriver _driver;
private String _host;
private int _port;
private String _database;
private Socket _s;
private MysqlReader _in;
private MysqlWriter _out;
private int _serverCapabilities;
private int _serverLanguage;
private byte []_scrambleBuf = new byte[8 + 13];
private byte []_errorState = new byte[5];
private String _catalog;
private State _state = State.IDLE;
enum State {
IDLE,
FIELD_HEADER,
FIELD_DATA,
};
MysqlConnectionImpl(QuercusMysqlDriver driver,
String url,
Properties info)
throws SQLException
{
if (driver == null)
throw new NullPointerException();
_driver = driver;
_host = driver.getHost();
_port = driver.getPort();
_database = driver.getDatabase();
_catalog = _database;
connect();
}
private void connect()
throws SQLException
{
Socket s = null;
try {
s = new Socket(_host, _port);
InputStream is = s.getInputStream();
OutputStream os = s.getOutputStream();
ReadStream in = Vfs.openRead(is);
WriteStream out = Vfs.openWrite(os);
MysqlReader reader = new MysqlReader(this, in);
MysqlWriter writer = new MysqlWriter(this, out);
readHandshakeInit(reader);
String user = "ferg";
String password = "";
writeClientAuth(writer, user, password, _database);
readOk(reader);
_s = s;
_in = reader;
_out = writer;
s = null;
} catch (IOException e) {
throw new SQLTransientConnectionException(L.l("{0}:{1} is not an accessible host",
_host, _port));
} finally {
if (s != null) {
try {
s.close();
} catch (Exception e) {
log.log(Level.FINEST, e.toString(), e);
}
}
}
}
/**
* JDBC api to create a new statement. Any SQL exception thrown here
* will make the connection invalid, i.e. it can't be put back into
* the pool.
*
* @return a new JDBC statement.
*/
public Statement createStatement()
throws SQLException
{
return new MysqlStatementImpl(this);
}
public Statement createStatement(int resultSetType, int resultSetConcurrency)
throws SQLException
{
return createStatement();
}
public String getCatalog()
throws SQLException
{
return _catalog;
}
/**
* Sets the JDBC catalog.
*/
public void setCatalog(String catalog)
throws SQLException
{
if (_catalog == catalog || _catalog != null && _catalog.equals(catalog))
return;
_catalog = catalog;
if (catalog == null)
return;
try {
writeQuery("use " + catalog);
readOk(_in);
} catch (IOException e) {
throw new SQLException(e);
}
}
/**
* Gets the connection's metadata.
*/
public DatabaseMetaData getMetaData()
throws SQLException
{
return new MysqlDatabaseMetaData(this);
}
public SQLWarning getWarnings()
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public void clearWarnings()
throws SQLException
{
}
public boolean getAutoCommit()
throws SQLException
{
return false;
}
public void setAutoCommit(boolean autoCommit)
throws SQLException
{
}
public void commit()
throws SQLException
{
}
public void rollback()
throws SQLException
{
}
/**
* Returns true if the connection is closed.
*/
public boolean isClosed()
throws SQLException
{
return false;
}
/**
* Reset the connection and return the underlying JDBC connection to
* the pool.
*/
public void close() throws SQLException
{
Socket s = _s;
_s = null;
MysqlWriter out = _out;
_out = null;
MysqlReader in = _in;
_in = null;
}
//
// packet methods
//
private void readHandshakeInit(MysqlReader in)
throws IOException, SQLException
{
in.readPacket();
int protocolVersion = in.readByte();
String serverVersion = in.readNullTermString();
int threadId = in.readInt();
in.readAll(_scrambleBuf, 0, 8);
in.skip(1);
_serverCapabilities = in.readShort();
_serverLanguage = in.readByte();
int serverStatus = in.readShort();
in.readAll(_scrambleBuf, 8, 13);
in.readAll(_scrambleBuf, 8, 13);
}
private String readOk(MysqlReader in)
throws IOException, SQLException
{
in.readPacket();
int fieldCount = in.readByte();
if (fieldCount == 0xff)
return readError(in);
if (fieldCount == 0x00) {
long rows = in.readLengthCodedBinary();
long insertId = in.readLengthCodedBinary();
int status = in.readShort();
int warningCount = in.readShort();
String message = in.readTailString();
in.endPacket();
return message;
}
else
throw new SQLException("unexpected field");
}
String readResult(MysqlResultImpl result)
throws SQLException
{
try {
MysqlReader in = _in;
in.readPacket();
int fieldCount = in.readByte();
if (fieldCount == 0xff)
return readError(in);
if (fieldCount == 0x00) {
result.setResultSet(false);
result.setUpdateCount((int) in.readLengthCodedBinary());
result.setInsertId(in.readLengthCodedBinary());
int status = in.readShort();
int warningCount = in.readShort();
String message = in.readTailString();
}
else if (fieldCount == 0xfe) {
result.setResultSet(false);
}
else {
result.setResultSet(true);
fieldCount = (int) in.readLengthCodedBinary(fieldCount);
readResultFields(result, fieldCount);
}
return null;
} catch (IOException e) {
throw new SQLException(e);
}
}
private void readResultFields(MysqlResultImpl result, int fieldCount)
throws IOException, SQLException
{
MysqlReader in = _in;
result.setColumnCount(fieldCount);
int index = 0;
while (true) {
in.readPacket();
int count = in.readByte();
if (count == 0xfe || count < 0) { // EOF
break;
}
MysqlColumn column = result.getColumn(index++);
if (count == 251) {
System.out.println("NULL:");
continue;
}
int len;
char []buffer;
len = (int) in.readLengthCodedBinary(count);
buffer = column.startCatalog(len);
in.readAll(buffer, 0, len);
len = (int) in.readLengthCodedBinary();
buffer = column.startDatabase(len);
in.readAll(buffer, 0, len);
len = (int) in.readLengthCodedBinary();
buffer = column.startTable(len);
in.readAll(buffer, 0, len);
len = (int) in.readLengthCodedBinary();
buffer = column.startOrigTable(len);
in.readAll(buffer, 0, len);
len = (int) in.readLengthCodedBinary();
buffer = column.startName(len);
in.readAll(buffer, 0, len);
len = (int) in.readLengthCodedBinary();
buffer = column.startOrigName(len);
in.readAll(buffer, 0, len);
in.readByte();
column.setCharset(in.readShort());
column.setLength(in.readInt());
column.setType(in.readByte());
column.setFlags(in.readShort());
column.setDecimals(in.readByte());
int filler = in.readShort();
// only for show tables
// column.setDefault(in.readLengthCodedBinary());
}
int warningCount = in.readShort();
int statusFlags = in.readShort();
result.setRowAvailable(true);
_state = State.FIELD_DATA;
}
private void skipRowData()
throws SQLException
{
if (_state != State.FIELD_DATA)
return;
try {
MysqlReader in = _in;
while (true) {
in.readPacket();
int count = in.readByte();
if (count == 0xfe || count < 0) { // EOF
_state = State.IDLE;
return;
}
}
} catch (IOException e) {
throw new SQLException(e);
}
}
boolean readRow(MysqlResultImpl result)
throws SQLException
{
assert(_state == State.FIELD_DATA);
try {
MysqlReader in = _in;
in.readPacket();
int count = in.readByte();
if (count == 0xfe || count < 0) { // EOF
_state = State.IDLE;
return false;
}
int fieldCount = result.getColumnCount();
TempOutputStream resultStream = result.getResultStream();
if (resultStream == null)
throw new NullPointerException();
int offset = 0;
int length = 0;
int index = 0;
MysqlColumn column = result.getColumn(index++);
length = (int) in.readLengthCodedBinary(count);
column.setRowOffset(offset);
column.setRowLength(length);
in.readData(resultStream, length);
offset += length;
for (fieldCount--; fieldCount > 0; fieldCount--) {
length = (int) in.readLengthCodedBinary();
column = result.getColumn(index++);
column.setRowOffset(offset);
column.setRowLength(length);
in.readData(resultStream, length);
offset += length;
}
return true;
} catch (IOException e) {
throw new SQLException(e);
}
}
private String readError(MysqlReader in)
throws IOException, SQLException
{
int errno = in.readShort();
int marker = in.readByte();
StringBuilder sb = new StringBuilder();
if (marker == '#') {
in.readAll(_errorState, 0, 5);
}
else
sb.append((char) marker);
int len = in.getPacketLength() - in.getPacketOffset();
for (int i = 0; i < len; i++) {
int ch = in.readByte();
if (ch > 0)
sb.append((char) ch);
}
System.out.println("ERROR: errno=" + errno + " stat:" + new String(_errorState) + " " + sb);
throw new SQLException(sb.toString());
}
private void writeClientAuth(MysqlWriter out,
String user,
String password,
String database)
throws IOException, SQLException
{
out.startVariablePacket();
int clientFlags = 0;
clientFlags |= CLIENT_PROTOCOL_41;
clientFlags |= CLIENT_LONG_PASSWORD;
clientFlags |= CLIENT_LONG_FLAG;
// clientFlags = 0x03a685;
clientFlags = 0x03a685;
clientFlags &= ~CLIENT_COMPRESS;
clientFlags &= ~CLIENT_SSL;
if (_database != null)
clientFlags |= CLIENT_CONNECT_WITH_DB;
out.writeInt(clientFlags);
int maxPacketSize = 1 << 24;
out.writeInt(maxPacketSize);
// int charsetNumber = UTF8_MB3;
int charsetNumber = 8;
out.writeByte(charsetNumber);
out.writeZero(23);
out.writeNullTermString(user);
byte []hash = new byte[0];
// out.writeNullTermString("");
// out.write(hash, 0, hash.length);
out.writeByte(0);
if (_database != null)
out.writeNullTermString(_database);
out.endVariablePacket();
out.flush();
}
void writeQuery(String query)
throws SQLException
{
if (_state != State.IDLE)
skipRowData();
if (log.isLoggable(Level.FINER))
log.finer(this + " query '" + query + "'");
try {
MysqlWriter out = _out;
int len = query.length() + 1;
out.writeByte(len);
out.writeByte(len >> 8);
out.writeByte(len >> 16);
out.writeByte(0); // id
out.writeByte(COM_QUERY);
out.write(query);
out.flush();
} catch (IOException e) {
throw new SQLException(e);
}
}
//
// stub methods - methods not used by Quercus mysql
//
public Statement createStatement(int resultSetType,
int resultSetConcurrency,
int resultSetHoldability)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public PreparedStatement prepareStatement(String sql)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public PreparedStatement prepareStatement(String sql,
int resultSetType,
int resultSetConcurrency)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public PreparedStatement prepareStatement(String sql,
int resultSetType,
int resultSetConcurrency,
int resultSetHoldability)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public PreparedStatement prepareStatement(String sql,
int resultSetType)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public PreparedStatement prepareStatement(String sql,
int []columnIndexes)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public PreparedStatement prepareStatement(String sql,
String []columnNames)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public CallableStatement prepareCall(String sql, int resultSetType,
int resultSetConcurrency)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public CallableStatement prepareCall(String sql)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public CallableStatement prepareCall(String sql,
int resultSetType,
int resultSetConcurrency,
int resultSetHoldability)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public Map getTypeMap()
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public void setTypeMap(Map> map)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public String nativeSQL(String sql)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public int getTransactionIsolation()
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public void setTransactionIsolation(int isolation)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public void setReadOnly(boolean readOnly)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public boolean isReadOnly()
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public void setHoldability(int hold)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public int getHoldability()
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public Savepoint setSavepoint()
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public Savepoint setSavepoint(String name)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public void releaseSavepoint(Savepoint savepoint)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public void rollback(Savepoint savepoint)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public Clob createClob()
throws SQLException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public Blob createBlob()
throws SQLException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public NClob createNClob()
throws SQLException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public SQLXML createSQLXML()
throws SQLException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public boolean isValid(int timeout)
throws SQLException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public void setClientInfo(String name, String value)
throws SQLClientInfoException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public void setClientInfo(Properties properties)
throws SQLClientInfoException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public String getClientInfo(String name)
throws SQLException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public Properties getClientInfo()
throws SQLException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public Array createArrayOf(String typeName, Object[] elements)
throws SQLException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public Struct createStruct(String typeName, Object[] attributes)
throws SQLException
{
throw new UnsupportedOperationException("Not supported yet.");
}
public T unwrap(Class iface)
throws SQLException
{
throw new UnsupportedOperationException(getClass().getName());
}
public boolean isWrapperFor(Class> iface)
throws SQLException
{
return false;
}
public void abort(Executor arg0) throws SQLException
{
}
public int getNetworkTimeout() throws SQLException
{
return 0;
}
public String getSchema() throws SQLException
{
return null;
}
public void setNetworkTimeout(Executor arg0, int arg1) throws SQLException
{
}
public void setSchema(String arg0) throws SQLException
{
}
public String toString()
{
return getClass().getSimpleName() + "[" + _host + ":" + _port + "]";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy