?? pgtcpredirect.java
字號:
/*
* Copyright 2004-2008 H2 Group. Licensed under the H2 License, Version 1.0
* (license2)
* Initial Developer: H2 Group
*/
package org.h2.tools.net;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
/**
* This class helps debug the PostgreSQL network protocol.
* It listens on one port, and sends the exact same data to another port.
*/
public class PgTcpRedirect {
public static void main(String[] args) throws Exception {
new PgTcpRedirect().loop(args);
}
void loop(String[] args) throws Exception {
// MySQL protocol:
// http://www.redferni.uklinux.net/mysql/MySQL-Protocol.html
// PostgreSQL protocol:
// http://developer.postgresql.org/pgdocs/postgres/protocol.html
// int portServer = 9083, portClient = 9084;
// int portServer = 3306, portClient = 3307;
// int portServer = 5435, portClient = 5433; // H2 PgServer
int portServer = 5432, portClient = 5433; // PostgreSQL
for (int i = 0; i < args.length; i++) {
if ("-client".equals(args[i])) {
portClient = Integer.parseInt(args[++i]);
} else if ("-server".equals(args[i])) {
portServer = Integer.parseInt(args[++i]);
}
}
ServerSocket listener = new ServerSocket(portClient);
while (true) {
Socket client = listener.accept();
Socket server = new Socket("localhost", portServer);
TcpRedirectThread c = new TcpRedirectThread(client, server, true);
TcpRedirectThread s = new TcpRedirectThread(server, client, false);
new Thread(c).start();
new Thread(s).start();
}
}
private class TcpRedirectThread implements Runnable {
private Socket read, write;
private int state;
private boolean client;
private static final int STATE_INIT_CLIENT = 0, STATE_REGULAR = 1;
TcpRedirectThread(Socket read, Socket write, boolean client) {
this.read = read;
this.write = write;
this.client = client;
state = client ? STATE_INIT_CLIENT : STATE_REGULAR;
}
String readStringNull(InputStream in) throws IOException {
StringBuffer buff = new StringBuffer();
while (true) {
int x = in.read();
if (x <= 0) {
break;
}
buff.append((char) x);
}
return buff.toString();
}
private void println(String s) {
// System.out.println(s);
}
private boolean processClient(InputStream inStream, OutputStream outStream) throws IOException {
DataInputStream dataIn = new DataInputStream(inStream);
ByteArrayOutputStream buff = new ByteArrayOutputStream();
DataOutputStream dataOut = new DataOutputStream(buff);
if (state == STATE_INIT_CLIENT) {
state = STATE_REGULAR;
int len = dataIn.readInt();
dataOut.writeInt(len);
len -= 4;
byte[] data = new byte[len];
dataIn.readFully(data, 0, len);
dataOut.write(data);
dataIn = new DataInputStream(new ByteArrayInputStream(data, 0, len));
int version = dataIn.readInt();
if (version == 80877102) {
println("CancelRequest");
println(" pid: " + dataIn.readInt());
println(" key: " + dataIn.readInt());
} else if (version == 80877103) {
println("SSLRequest");
} else {
println("StartupMessage");
println(" version " + version + " (" + (version >> 16) + "." + (version & 0xff) + ")");
while (true) {
String param = readStringNull(dataIn);
if (param.length() == 0) {
break;
}
String value = readStringNull(dataIn);
println(" param " + param + "=" + value);
}
}
} else {
int x = dataIn.read();
if (x < 0) {
println("end");
return false;
}
// System.out.println(" x=" + (char)x+" " +x);
dataOut.write(x);
int len = dataIn.readInt();
dataOut.writeInt(len);
len -= 4;
byte[] data = new byte[len];
dataIn.readFully(data, 0, len);
dataOut.write(data);
dataIn = new DataInputStream(new ByteArrayInputStream(data, 0, len));
switch (x) {
case 'B': {
println("Bind");
println(" destPortal: " + readStringNull(dataIn));
println(" prepName: " + readStringNull(dataIn));
int formatCodesCount = dataIn.readShort();
for (int i = 0; i < formatCodesCount; i++) {
println(" formatCode[" + i + "]=" + dataIn.readShort());
}
int paramCount = dataIn.readShort();
for (int i = 0; i < paramCount; i++) {
int paramLen = dataIn.readInt();
println(" length[" + i + "]=" + paramLen);
byte[] d2 = new byte[paramLen];
dataIn.readFully(d2);
}
int resultCodeCount = dataIn.readShort();
for (int i = 0; i < resultCodeCount; i++) {
println(" resultCodeCount[" + i + "]=" + dataIn.readShort());
}
break;
}
case 'C': {
println("Close");
println(" type: (S:prepared statement, P:portal): " + dataIn.read());
break;
}
case 'd': {
println("CopyData");
break;
}
case 'c': {
println("CopyDone");
break;
}
case 'f': {
println("CopyFail");
println(" message: " + readStringNull(dataIn));
break;
}
case 'D': {
println("Describe");
println(" type (S=prepared statement, P=portal): " + (char) dataIn.readByte());
println(" name: " + readStringNull(dataIn));
break;
}
case 'E': {
println("Execute");
println(" name: " + readStringNull(dataIn));
println(" maxRows: " + dataIn.readShort());
break;
}
case 'H': {
println("Flush");
break;
}
case 'F': {
println("FunctionCall");
println(" objectId:" + dataIn.readInt());
int columns = dataIn.readShort();
for (int i = 0; i < columns; i++) {
println(" formatCode[" + i + "]: " + dataIn.readShort());
}
int count = dataIn.readShort();
for (int i = 0; i < count; i++) {
int l = dataIn.readInt();
println(" len[" + i + "]: " + l);
if (l >= 0) {
for (int j = 0; j < l; j++) {
dataIn.readByte();
}
}
}
println(" resultFormat: " + dataIn.readShort());
break;
}
case 'P': {
println("Parse");
println(" name:" + readStringNull(dataIn));
println(" query:" + readStringNull(dataIn));
int count = dataIn.readShort();
for (int i = 0; i < count; i++) {
println(" [" + i + "]: " + dataIn.readInt());
}
break;
}
case 'p': {
println("PasswordMessage");
println(" password: " + readStringNull(dataIn));
break;
}
case 'Q': {
println("Query");
println(" sql : " + readStringNull(dataIn));
break;
}
case 'S': {
println("Sync");
break;
}
case 'X': {
println("Terminate");
break;
}
default:
println("############## UNSUPPORTED: " + (char) x);
}
}
dataOut.flush();
byte[] buffer = buff.toByteArray();
printData(buffer, buffer.length);
try {
outStream.write(buffer, 0, buffer.length);
outStream.flush();
} catch (IOException e) {
e.printStackTrace();
}
return true;
}
private boolean processServer(InputStream inStream, OutputStream outStream) throws IOException {
DataInputStream dataIn = new DataInputStream(inStream);
ByteArrayOutputStream buff = new ByteArrayOutputStream();
DataOutputStream dataOut = new DataOutputStream(buff);
int x = dataIn.read();
if (x < 0) {
println("end");
return false;
}
// System.out.println(" x=" + (char)x+" " +x);
dataOut.write(x);
int len = dataIn.readInt();
dataOut.writeInt(len);
len -= 4;
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -