首页 > 开发 > 综合 > 正文

Visual C#实现HTTP代理服务程序

2024-07-21 02:29:37
字体:
来源:转载
供稿:网友
  网络代理程序的种类非常多,根据代理服务程序代理的协议不同,分成http代理服务程序、ftp代理服务程序等,运行代理服务程序的服务器也就称为http代理服务器和ftp代理服务器。在本节中介绍的web代理服务程序代理的就是http协议。

  一.网络代理的类型及实现原理:

  网络代理服务根据工作层次,一般可分为应用层代理、传输层代理和socks代理。应用层代理是工作在tcp/ip参考模型的应用层之上,它支持对应用层协议(如http,ftp)的代理。它提供的控制最多,但是不灵活,必须要有相应的协议支持。如果协议不支持代理(如smtp和pop),那就只能在应用层以下代理,也即传输层代理。传输层代理直接与tcp层交互,更加灵活。要求代理服务器具有部分真正服务器的功能:监听特定tcp或udp端口,接收客户端的请求同时向客户端发出相应的响应。另一种代理需要改变客户端的ip栈,即socks代理。它是可用的最强大、最灵活的代理标准协议。sock v4允许代理服务器内部的客户端完全地连接到外部的服务器,sock v5增加了对客户端的授权和认证,因此它是一种安全性较高的代理。本节后面介绍的代理是一种应用层上面的代理,所代理的协议是http,也就是经常见到的web代理。

  网络代理就是一个连接客户端(设定需要代理的计算机)和服务器端(需要访问资源的服务器)的桥。要实现这种桥,网络代理就必须满足下列条件,其实也是代理服务的运行的流程:

  (1). 能够接收并解析客户端的请求。

  (2). 创建到服务器的新连接,并根据转发客户端的请求信息。

  (3). 接收服务器反馈的信息。

  (4). 能够发出或解释服务器的响应并将该响应传回给客户端。

  图01是网络代理服务的一个典型模型图:


图01:代理服务的模型

  二.visual c#实现web代理服务程序

  web代理服务是代理服务中最常用的一种代理服务,按照代理服务的层次,它属于应用层代理,是对tcp/ip参考模型中的应用层的http协议的代理。

  web代理服务也是代理服务中的一种,所以它也要满足代理服务的基本条件。在下面介绍的代理服务程序中,是按照下列的顺序来实现其功能的。

  (1). 代理服务器程序侦听端口,接收客户端浏览器发送来的web请求信息。

  (2). 代理服务器程序接收到客户端web请求信息后,解析出web服务器的地址,并创建一个socket实例,并以此实例连接web服务器上。

  (3). 通过创建的socket传送客户端的web请求数据包到web服务器的80端口。

  (4). 代理服务器程序接收web服务器返回页面数据。

  (5). 代理服务器程序把接收来的数据传送到客户端,实现web代理。

  由于客户端的对一个地址的浏览,要传送很多的web请求信息,为了更快、更准确的处理这些信息,web代理服务程序采用了多线程来处理每一个web请求。细心的读者可能会发现,处理每一个客户端的web请求信息,代理服务器软件都要使用二个socket,一个是用来接收/传送客户机的信息,一个是和web服务器进行交流。为了区分这二个socket,我们把他们都命名,和服务器对话的socket,称为服务socket;和客户端机器对话的scoket,称为客户socket。

  下面就开始web代理服务程序的编写工作。

  这个示例主要包含三个部分内容:

  ·创建一个web代理类。

  ·web代理服务的类的实例化。

  ·如何通过这个web代理类的实例实现web代理服务。

  下面就是第一部分的具体的实现步骤。

  (一).创建一个web代理类

  以下是具体的操作步骤如下:

  1. 首先启动visual studio .net,依次选择"文件"、"新建"、"项目"菜单后,在弹出"新建项目"对话框中将"项目类型"设置为"visual c#项目",将"模板"设置为"windows应用程序",在"名称"文本框中输入"webproxy",在"位置"的文本框中输入"e:/vs.net项目",然后单击"确定"按钮。这样在"e:/vs.net项目"目录中就创建了一个新名称为"webproxy"文件夹,里面存放的就是"webproxy"的项目文件。

  2. 选择菜单【项目】|【添加类】,弹出【添加新项】对话框

  3. 将【模板】设置【类】

  4. 在【名称】文本框中输入【proxy】,单击【打开】按钮,具体如图02所示。


图02:web代理项目中【添加新项】对话框

  5. 在【解决方案资源管理器】窗口中,双击proxy.cs文件,进入proxy.cs文件的编辑界面。

  6. 在proxy.cs源文件的开头,添加下列代码,下列代码是导入proxy.cs中要使用到的命名空间:

using system ;
using system.net ;
using system.net.sockets ;
using system.text ;
using system.io ;
  7. 用下列构造函数替代默认的构造函数。下面的代码是在proxy类中创建一个构造函数。 proxy类只有一个构造函数,并且这个构造函数只有一个参数,这个参数是socket对象,它主要用来和客户端进行数据交换,是一个客户socket.。

public proxy ( socket socket )
{
 //
 // todo: 在此处添加构造函数逻辑
 //
 this.clientsocket = socket ;
}

  8. 创建proxy类中的run方法,run方法是proxy类中唯一的方法。其功能是从客户端接收http请求,并传送到web服务器,然后从web服务器接收反馈来的数据,并传送到客户端。为了实现这二个不同方面的数据传送,run方法中是通过二个socket实例来实现的。在编写run方法的时候,要注意下面几点:

  (1). 由于http是tcp/ip参考模型中的应用层协议,它建立于tcp协议之上,所以创建的socket实例使用的协议类型应该为tcp协议。下面代码是创建可以传送http请求命令到web服务器和接收来自web服务器反馈来信息的socket实例:

socket ipsocket = new socket(addressfamily.internetwork, sockettype.stream,protocoltype.tcp);

  (2). 另外一个socket是在代理服务程序侦听端口号,接收挂起的连接请求时候得到的,以此socket为参数,利用proxy类中的构造函数,来创建一个proxy实例的。此socket实现从客户端接收http请求信息,并传送数据到客户端。

  socket创建和使用是实现web代理软件的关键,具体实现方法是在构造函数代码后面,输入下列代码,创建proxy类的run方法:

public void run ( )
{
 string clientmessage = " " ;
 //存放来自客户端的http请求字符串
 string url = " " ;
 //存放解析出地址请求信息
 int bytes = readmessage ( read , ref clientsocket , ref clientmessage ) ;
 if ( bytes == 0 )
 {
  return ;
 }

 int index1 = clientmessage.indexof ( ' ' ) ;
 int index2 = clientmessage.indexof ( ' ' , index1 + 1 ) ;
 if ( ( index1 == -1 ) || ( index2 == -1 ) )
 {
  throw new ioexception ( ) ;
 }
 string part1 = clientmessage.substring ( index1 + 1 , index2 - index1 ) ;
 int index3 = part1.indexof ( '/' , index1 + 8 ) ;
 int index4 = part1.indexof ( ' ' , index1 + 8 ) ;
 int index5 = index4 - index3 ;
 url = part1.substring ( index1 + 4 , ( part1.length - index5 ) - 8 ) ;

 try
 {
  iphostentry iphost = dns.resolve ( url ) ;
  console.writeline ( "远程主机名: " + iphost.hostname ) ;
  string [] aliases = iphost.aliases ;
  ipaddress[] address = iphost.addresslist ;
  console.writeline ( "web服务器ip地址:" + address[0] ) ;
  //解析出要访问的服务器地址
  ipendpoint ipendpoint = new ipendpoint ( address[0] , 80 ) ;
  socket ipsocket = new socket ( addressfamily.internetwork , sockettype.stream , protocoltype.tcp ) ;
  //创建连接web服务器端的socket对象
  ipsocket.connect ( ipendpoint ) ;
  //socket连web接服务器
  if ( ipsocket.connected )
   console.writeline ( "socket 正确连接!" ) ;
  string get = clientmessage ;
  byte[] byteget = ascii.getbytes ( get ) ;
  ipsocket.send ( byteget , byteget.length , 0 ) ;
  //代理访问软件对服务器端传送http请求命令
  int32 rbytes = ipsocket.receive ( recvbytes , recvbytes.length , 0 ) ;
  //代理访问软件接收来自web服务器端的反馈信息
  console.writeline ( "接收字节数:" + rbytes.tostring ( ) ) ;
  string strretpage = null ;
  strretpage = strretpage + ascii.getstring ( recvbytes , 0 , rbytes ) ;
  while ( rbytes > 0 )
  {
   rbytes = ipsocket.receive ( recvbytes , recvbytes.length , 0 ) ;
   strretpage = strretpage + ascii.getstring ( recvbytes , 0 , rbytes ) ;
  }
  ipsocket.shutdown ( socketshutdown.both ) ;
  ipsocket.close ( ) ;
  sendmessage ( clientsocket , strretpage ) ;
  //代理服务软件往客户端传送接收到的信息
 }
 catch ( exception exc2 )
 {
  console.writeline ( exc2.tostring ( ) ) ;
 }
}
//接收客户端的http请求数据
private int readmessage ( byte [ ] bytearray , ref socket s , ref string clientmessage )
{
 int bytes = s.receive ( bytearray , 1024 , 0 ) ;
 string messagefromclient = encoding.ascii.getstring ( bytearray ) ;
 clientmessage = ( string )messagefromclient ;
 return bytes ;
}
//传送从web服务器反馈的数据到客户端
private void sendmessage ( socket s , string message )
{
 buffer = new byte[message.length + 1] ;
 int length = ascii.getbytes ( message , 0 , message.length , buffer , 0 ) ;
 console.writeline ( "传送字节数:" + length.tostring ( ) ) ;
 s.send ( buffer , length , 0 ) ;
}

  9. 在定义proxy类代码区中加入下列代码,下列代码是定义proxy类中的使用的一些变量,这些变量主要是在后面的定义run方法中使用。

socket clientsocket ;
byte[] read = new byte[1024] ;
//定义一个空间,存储来自客户端请求数据包
byte [] buffer = null ;
encoding ascii = encoding.ascii ;
//设定编码
byte[] recvbytes = new byte[4096] ;
//定义一个空间,存储web服务器返回的数据

  10. 至此,proxy类的定义过程就完成了。把proxy类实例化非常简单,和以前用的其他完全一样,具体语法如下:

public proxy ( socket socket );

  参数:socket为一个scoket实例

  下面代码是创建一个proxy实例:

proxy proxy = new proxy ( socket ) ;
  (二). 利用proxy类,实现web代理的具体示例:

  下面是利用上面创建的proxy类,实现web代理程序的具体实现步骤,proxy类被定义在命名空间webproxy中。

  1. 在visual studio .net的代码编辑器中打开class1.cs文件,进入class1.cs的代码编辑界面。

  2. 在class1.cs源文件的开头导入下列命名空间:

using system ;
using system.net ;
using system.net.sockets ;
using system.text ;
using system.io ;
using system.threading ;
using webproxy ;
//其中命名空间webproxy是proxy类所处的位置,具体可以参阅proxy.cs源文件
//中命名空间的定义。

  3. 在main函数中添加下列代码,下列代码是利用proxy类,来实现web代理程序。

const int port = 8000 ;
//定义端口号
tcplistener tcplistener = new tcplistener ( port ) ;
console.writeline ( "侦听端口号: " + port.tostring ( ) ) ;
tcplistener.start ( ) ;
//侦听端口号
while ( true )
{
 socket socket = tcplistener.acceptsocket ( ) ;
 //并获取传送和接收数据的scoket实例
 proxy proxy = new proxy ( socket ) ;
 //proxy类实例化
 thread thread = new thread ( new threadstart ( proxy.run ) ) ;
 //创建线程
 thread.start ( ) ;
 //启动线程
}

  保存上面的所有步骤,这样一个简单web代理程序就算是完成了。此web代理程序侦听的是8000端口号。

  (三).测试web代码程序:

  web代理程序要通过二台计算机才能够实现。其中的一台计算机运行web代理程序,充当web代理服务器。另外一台计算机充当客户机,通过web代理服务器来浏览网页。在确定web代理软件运行后,下面是对客户机进行必要的设置。

  1. 打开ie浏览器。

  2. 选择【工具】|【internet选项】,弹出【internet选项】对话框。在此对话框中选择【连接】页面,单击其中的【局域网设置】按钮。弹出【局域网(lan)设置】对话框。选择【为lan使用代理服务器(x),(这些设置不会应用于拨号和vpn连接)】多选框。并在其中的【地址】文本框中输入代理服务器的ip地址,由于测试的代理服务器的ip地址为"10.138.198.213",所有也输入此ip地址,在【端口】文本框中输入"8000"。具体如图03所示:


图03:客户端设定web代理服务器对话框

  此时客户端的设置就完成了,在确定ip地址为"10.138.198.213"的这台计算机已经运行上面介绍的web代理程序后。打开客户端的ie浏览器,并输入要浏览的网址,就可以通过web代理服务器来浏览网页了,图04是web代理服务程序在服务器端运行时的界面。


图04:web代理服务程序在服务器端的运行界面

  四.总结

  至此一个简单的web代理服务软件就算基本完成了,通过上面内容的介绍可见,虽然代理服务的实现原理相对简单,但具体实现其实还是很繁琐的。网络代理是一个内容丰富,实现复杂的论题,本节介绍的代理服务软件,无论在实现的协议种类,还是实现的功能,都只能算很小的一部分。希望各位能够通过本文的介绍,结合其他相关的知识,创造出功能更强大、安全性更高,使用更稳定的网络代理服务程序来。
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表