使用socket实现的请求代理

 

    前段时间把TCP/UDP协议复习了一下,了解了一般网络编程的基本步骤,以前在工作中遇到以下两个问题,就可以使用socket代理的方式实现:

 

1.和化为10000号调接口的时候,我们是java web 系统,他们是vc++ 开发的系统,我们对10000号提供webservice服务,联调的时候就比较麻烦,我们走的是http,向一个wsdl地址发送String,然后接受服务器返回的String,相同的入参字符串,给了10000号系统,他们调用就总报错,因为服务器端截获不到报文,我无法定位错误原因,我们的web 系统使用的是 axis 实现的webservice,在地层已经帮助我们封装好了 soap 协议的报文,而10000号系统使用的是socket,如何获得10000号系统的请求报文的底层呢??就需要我们使用 一下的一个 socket代理

 

                                             c++ ---------------- <- ----------- java

                      10000号系统 ----------------> Scoket代理------------------>目标服务器

 

 

也就是我们使用socket 代理截获底层报文,然后又代理吧报文发给目标服务器,然后socket代理接受目标服务器的报文,返回给10000号系统,这个时候我们就有了 请求和返回的报文,把它以文件的形式存在硬盘上。然后分析报文。

 

2.和ismp联调的时候他们给了wsdl的发布地址,http://133.64.64.13:5011/IsmpCrmEngineService?wsdl,我本地开发就是ping不通,自然也钓不上他们的服务,估计是路由问题,可是我们的服务器 133.64.80.9 可以ping通这个

地址,于是我们也可以使用socket代理,我们在133.64.80.9 部署socket代理 ,我们去请求socket代理,让 代理去请求 目标地址

 

3.socket 代理 还有一个好处就比较牛逼了,我们的系统需要调用tuxedo系统的服务,走的也是socket,比如你现在家,你boss打电话说生产系统 页面上不能查话费了,查话费走的是tuxedo,然你去电信看看,找找原因,你当然不愿意

不愿意飙到电信,可是你如何知道是接口问题还是别的问题呢?? 你在家里的网络环境如何调用电信DCN网络环境的tuxedo服务呢?嘿嘿,我们生产环境 是一个web系统 在公网上有一个ip地址,你可以把 Scoket代理部署到生产环境上,然后你在家里的 公网上调用 socket代理 ,然代理去调用 DCN网内部的服务,把结果从电信的DCN 网络环境中的

请求结果报文返回给公网上的你。(请勿模仿,后果自负

 

 

下面我们就根据以上的构想,开发一个socket代理:(知道tcp/udp协议和编程技巧),不明白的可以看我的

 

tcp编程:http://8366.iteye.com/admin/blogs/419331

udp编程:http://8366.iteye.com/admin/blogs/420395

 

代码:

 

 

1.    写缓存文件

 

import java.io.FileOutputStream;
import java.io.IOException;

import org.apache.log4j.Logger;

/**
 * 写缓存文件
 *  *
 */
public class FileWrite
{
	private static Logger log = Logger.getLogger(FileWrite.class);
	
	/** 文件编号 */
	static int file_no = 0;
	
	/**
	 * 写文件
	 * @param buff 内容
	 * @param len 长度
	 * @param type 文件后缀
	 * @throws Exception
	 */
	public static void writeFile(byte[] buff, int len, String type)
	{
		file_no++;
		
		//文件路径
		StringBuilder fileNo = new StringBuilder(SocketProxy.OUT_PATH);
		//文件名编号3位对齐
		for (int i = 0; i < 3 - ("" + file_no).length(); i++)
		{
			fileNo.append("0");
		}
		//文件编号
		fileNo.append(file_no);
		fileNo.append("_");
		//文件后缀
		fileNo.append(type);
		
		FileOutputStream out = null;
		try
		{
			out = new FileOutputStream(fileNo.toString());
			out.write(buff, 0, len);
		}
		catch (IOException e)
		{
			log.error("写文件异常", e);
		}
		finally
		{
			try
			{
				out.close();
			}
			catch (IOException e)
			{
			}
		}
	}
}

 

 2.请求线程

 

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;

import org.apache.log4j.Logger;


public class SubRequest extends Thread
{
	private static Logger log = Logger.getLogger(SubRequest.class);

	/** 外部socket */
	Socket outSideSocket;
	/** 外部输入 */
	InputStream in;
	
	/** 内部socket */
	Socket inSideSocket;
	/** 向服务端输出 */
	OutputStream out;
	
	/** 外部输入缓冲区 */
	byte[] buff = new byte[SocketProxy.OUT_INPUT_BUF_LENGTH];
	
	public SubRequest(Socket outSideSocket, Socket inSideSocket)throws Exception
	{
		this.outSideSocket = outSideSocket;
		this.inSideSocket = inSideSocket;
		
		in = outSideSocket.getInputStream();		
		out = inSideSocket.getOutputStream();
	}
	
	public void run()
	{
		while(true)
		{
			try
			{
				int n = in.read(buff);
				log.info("输入字节数:" + n);
				
				//输入字节长度-1表示连接关闭
				if (n == -1)
				{
					break;
				}
				
				//写文件
				FileWrite.writeFile(buff, n, "request");
				//输出
				out.write(buff, 0, n);
			}
			catch (Exception e)
			{
				log.error("输入异常", e);
				break;
			}
		}
		
		//关闭socket
		if (!outSideSocket.isClosed())
		{
			try
			{
				in.close();
				outSideSocket.close();
				log.info("关闭外部socket");
			}
			catch (IOException ex)
			{						
			}
		}
		if (!inSideSocket.isClosed())
		{
			try
			{
				out.close();
				inSideSocket.close();
				log.info("关闭内部socket");
			}
			catch (IOException ex)
			{						
			}
		}

	}
	

}

 

3.接受线程

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;

import org.apache.log4j.Logger;


public class SubResponse extends Thread
{
	private static Logger log = Logger.getLogger(SubResponse.class);
	
	/** 外部socket */
	Socket outSideSocket;
	/** 向客户端输出 */
	OutputStream out;
	
	/** 内部socket */
	Socket inSideSocket;
	/** 服务端输入 */
	InputStream in;
	
	/** 内部发送缓冲区 */
	byte[] buff = new byte[SocketProxy.IN_OUTPUT_BUF_LENGTH];

	
	public SubResponse(Socket outSideSocket, Socket inSideSocket)throws Exception
	{
		this.outSideSocket = outSideSocket;
		out = outSideSocket.getOutputStream();
		
		this.inSideSocket = inSideSocket;
		in = inSideSocket.getInputStream();
	}
	
	public void run()
	{
		while (true)
		{
			try
			{
				int n = in.read(buff);
				log.info("输出字节数:" + n);
				
				//输入字节长度-1表示连接关闭
				if (n == -1)
				{
					break;
				}
				
				//写文件
				FileWrite.writeFile(buff, n, "response");
				//输出
				out.write(buff, 0, n);
			}
			catch (Exception e)
			{
				log.error("输出异常", e);
				break;
			}
		}
		
		//关闭socket
		if (!outSideSocket.isClosed())
		{
			try
			{
				out.close();
				outSideSocket.close();
				log.info("关闭外部socket");
			}
			catch (IOException ex)
			{						
			}
		}
		if (!inSideSocket.isClosed())
		{
			try
			{
				in.close();
				inSideSocket.close();
				log.info("关闭内部socket");
			}
			catch (IOException ex)
			{						
			}
		}
	}
}

 

 

4.主线程

import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;

import org.apache.log4j.Logger;


public class SocketProxy
{
	private static Logger log = Logger.getLogger(SocketProxy.class);

	/** 代理对外侦听端口 */
	static int port = 9999;

	/** 内部服务端IP  */
	static String ip = "133.64.41.134";
	/** 内部服务端端口  */
	static int in_port =8299;

	/** 客户端输入缓冲区长度 */
	static final int OUT_INPUT_BUF_LENGTH = 2048 * 10;
	/** 服务端输出缓冲区长度 */
	static final int IN_OUTPUT_BUF_LENGTH = 4096 * 10;

	/** 侦听报文输出目录 */
	static final String OUT_PATH = "c:/";

	/** 代理对外侦听Socket */
	ServerSocket serverSocket;

	public static void main(String[] args)
	{
		SocketProxy socketProxy = new SocketProxy();

		log.info("服务侦听:" + port);

		socketProxy.run();
	}

	public SocketProxy()
	{
		try
		{
			//侦听
			serverSocket = new ServerSocket(port);
		}
		catch (IOException e)
		{
			log.error("Server bind port " + port + " ERR.", e);
			System.exit(-1);
		}
	}

	public void run()
	{
		while(true)
		{
			try
			{
				Socket socket = serverSocket.accept();

				log.debug("接收到1个外部连接请求");

				//连接内部服务
				Socket inSideSocket = new Socket(ip, in_port);

				//启动Request子线程
				new SubRequest(socket, inSideSocket).start();
				//启动Response子线程
				new SubResponse(socket, inSideSocket).start();
			}
			catch (Throwable e)
			{
				log.warn("socket异常,退出", e);
				System.exit(-1);
			}
		}
	}
}

 

 

 

 使用方法:

 

在SocketProxy.java 中配置

 /** 代理对外侦听端口,也就是你的socket代理监听那个端口的消息*/
 static int port = 9999;

 /** 内部服务端IP,也就是收到请求以后,转发给哪个目标ip地址  */
 static String ip = "133.64.64.13";
 /** 内部服务端端口 ,目标地址的端口号 */
 static int in_port =5011;

 /*请求和相应报文在硬盘上的存放位置*/

static final String OUT_PATH = "c:/";

运行SocketProxy.java

 

然后我们请求的使用 只需要讲请求的 目标地址换成代理Socket的 ip 地址和它监听的端口号就可以了

 

 

附件描述:

 

附件中是 一个Socket代理,改了个名字 叫HttpAdapter, 写了个shell,可以部署在linux ,unix上,你只要改改配置文件就可以了

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值