分享
 
 
 

VC++6实现简单的代理服务器

王朝vc·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

#include "stdafx.h"

#include "Proxy.h"

#include < winsock2.h > //WINSOCKET API 2。0

#include < stdlib.h >

#include < stdio.h >

#include < string.h >

#ifdef _DEBUG

#define new DEBUG_NEW

#undef THIS_FILE

static char THIS_FILE[] = __FILE__;

#endif

/////////////////////////////////////////////////////////////////////////////

#define HTTP "http://"

#define FTP "ftp://"

#define PROXYPORT 5001 //Proxy 端口

#define BUFSIZE 10240 //缓冲区大小

CWinApp theApp;

using namespace std;

UINT ProxyToServer(LPVOID pParam);

UINT UserToProxyThread(void *pParam);

struct SocketPair{

SOCKET user_proxy; //socket : 本地机器到PROXY 服务机

SOCKET proxy_server; //socket : PROXY 服务机到远程主机

BOOL IsUser_ProxyClosed; // 本地机器到PROXY 服务机状态

BOOL IsProxy_ServerClosed; // PROXY 服务机到远程主机状态

};

struct ProxyParam{

char Address[256]; // 远程主机地址

HANDLE User_SvrOK; // PROXY 服务机到远程主机的联结状态

SocketPair *pPair; // 维护一组SOCKET的指针

int Port; // 用来联结远程主机的端口

}; //这个结构用来PROXY SERVER与远程主机的信息交换.

SOCKET gListen_Socket; //用来侦听的SOCKET。

int StartServer() //启动服务

{

WSADATA wsaData;

sockaddr_in local;

SOCKET listen_socket;

if(::WSAStartup(0x202,&wsaData)!=0)

{printf("\nError in Startup session.\n");WSACleanup();return -1;};

local.sin_family=AF_INET;

local.sin_addr.s_addr=INADDR_ANY;

local.sin_port=htons(PROXYPORT);

listen_socket=socket(AF_INET,SOCK_STREAM,0);

if(listen_socket==INVALID_SOCKET)

{printf("\nError in New a Socket.");WSACleanup();return -2;}

if(::bind(listen_socket,(sockaddr *)&local,sizeof(local))!=0)

{printf("\n Error in Binding socket."); WSACleanup();return -3; };

if(::listen(listen_socket,5)!=0)

{printf("\n Error in Listen."); WSACleanup(); return -4;}

gListen_Socket=listen_socket;

AfxBeginThread(UserToProxyThread,NULL); //启动侦听

return 1;

}

int CloseServer() //关闭服务

{

closesocket(gListen_Socket);

WSACleanup();

return 1;

}

//分析接收到的字符,得到远程主机地址

int GetAddressAndPort( char * str, char *address, int * port)

{

char buf[BUFSIZE], command[512], proto[128], *p;

int j;

sscanf(str,"%s%s%s",command,buf,proto);

p=strstr(buf,HTTP);

//HTTP

if(p)

{

p+=strlen(HTTP);

for(int i=0;i< strlen(p);i++)

if( *(p+i)=='/') break;

*(p+i)=0;

strcpy(address,p);

p=strstr(str,HTTP);

for(int j=0;j< i+strlen(HTTP);j++)

*(p+j)=' '; //去掉远程主机名: GET http:/www.csdn.net/ HTTP1.1 == > GET / HTTP1.1

*port=80; //缺省的 http 端口

}

else

{//FTP, 不支持, 下面的代码可以省略.

p=strstr(buf,FTP);

if(!p) return 0;

p+=strlen(FTP);

for(int i=0;i< strlen(p);i++)

if( *(p+i)=='/') break; //Get The Remote Host

*(p+i)=0;

for(j=0;j< strlen(p);j++)

if(*(p+j)==':')

{*port=atoi(p+j+1); //Get The Port

*(p+j)=0;

}

else *port=21;

strcpy(address,p);

p=strstr(str,FTP);

for(j=0;j< i+strlen(FTP);j++)

*(p+j)=' ';

}

return 1;

}

// 取到本地的数据,发往远程主机

UINT UserToProxyThread(void *pParam)

{

char Buffer[BUFSIZE];

int Len;

sockaddr_in from;

SOCKET msg_socket;

int fromlen,retval;

SocketPair SPair;

ProxyParam ProxyP;

CWinThread *pChildThread;

fromlen=sizeof(from);

msg_socket=accept(gListen_Socket,(struct sockaddr*)&from,&fromlen);

AfxBeginThread(UserToProxyThread,pParam); //启动另一侦听.

if( msg_socket==INVALID_SOCKET)

{ printf( "\nError in accept "); return -5;}

//读客户的第一行数据

SPair.IsUser_ProxyClosed=FALSE;

SPair.IsProxy_ServerClosed=TRUE;

SPair.user_proxy=msg_socket;

retval=recv(SPair.user_proxy,Buffer,sizeof(Buffer),0);

if(retval==SOCKET_ERROR)

{ printf("\nError Recv");

if(SPair.IsUser_ProxyClosed==FALSE)

{closesocket(SPair.user_proxy);

SPair.IsUser_ProxyClosed=TRUE;

}

}

if(retval==0)

{printf("Client Close connection\n");

if(SPair.IsUser_ProxyClosed==FALSE)

{closesocket(SPair.user_proxy);

SPair.IsUser_ProxyClosed=TRUE;

}

}

Len=retval;

#ifdef _DEBUG

Buffer[Len]=0;

printf("\n Received %d bytes,data[%s]from client\n",retval,Buffer);

#endif

//

SPair.IsUser_ProxyClosed=FALSE;

SPair.IsProxy_ServerClosed=TRUE;

SPair.user_proxy=msg_socket;

ProxyP.pPair=&SPair;

ProxyP.User_SvrOK=CreateEvent(NULL,TRUE,FALSE,NULL);

GetAddressAndPort( Buffer,ProxyP.Address,&ProxyP.Port);

pChildThread=AfxBeginThread(ProxyToServer,(LPVOID)&ProxyP);

::WaitForSingleObject(ProxyP.User_SvrOK,60000); //等待联结

::CloseHandle(ProxyP.User_SvrOK);

while(SPair.IsProxy_ServerClosed ==FALSE && SPair.IsUser_ProxyClosed==FALSE)

{

retval=send(SPair.proxy_server,Buffer,Len,0);

if(retval==SOCKET_ERROR)

{ printf("\n send() failed:error%d\n",WSAGetLastError());

if(SPair.IsProxy_ServerClosed==FALSE)

{

closesocket(SPair.proxy_server);

SPair.IsProxy_ServerClosed=TRUE;

}

continue;

}

retval=recv(SPair.user_proxy,Buffer,sizeof(Buffer),0);

if(retval==SOCKET_ERROR)

{ printf("\nError Recv");

if(SPair.IsUser_ProxyClosed==FALSE)

{closesocket(SPair.user_proxy);

SPair.IsUser_ProxyClosed=TRUE;

}

continue;

}

if(retval==0)

{printf("Client Close connection\n");

if(SPair.IsUser_ProxyClosed==FALSE)

{closesocket(SPair.user_proxy);

SPair.IsUser_ProxyClosed=TRUE;

}

break;

}

Len=retval;

#ifdef _DEBUG

Buffer[Len]=0;

printf("\n Received %d bytes,data[%s]from client\n",retval,Buffer);

#endif

} //End While

if(SPair.IsProxy_ServerClosed==FALSE)

{

closesocket(SPair.proxy_server);

SPair.IsProxy_ServerClosed=TRUE;

}

if(SPair.IsUser_ProxyClosed==FALSE)

{closesocket(SPair.user_proxy);

SPair.IsUser_ProxyClosed=TRUE;

}

::WaitForSingleObject(pChildThread- >m_hThread,20000); //Should check the return value

return 0;

}

// 读取远程主机数据,并发往本地客户机

UINT ProxyToServer(LPVOID pParam){

ProxyParam * pPar=(ProxyParam*)pParam;

char Buffer[BUFSIZE];

char *server_name= "localhost";

unsigned short port ;

int retval,Len;

unsigned int addr;

int socket_type ;

struct sockaddr_in server;

struct hostent *hp;

SOCKET conn_socket;

socket_type = SOCK_STREAM;

server_name = pPar- >Address;

port = pPar- >Port;

if (isalpha(server_name[0])) { /* server address is a name */

hp = gethostbyname(server_name);

}

else { /* Convert nnn.nnn address to a usable one */

addr = inet_addr(server_name);

hp = gethostbyaddr((char *)&addr,4,AF_INET);

}

if (hp == NULL ) {

fprintf(stderr,"Client: Cannot resolve address [%s]: Error %d\n",

server_name,WSAGetLastError());

::SetEvent(pPar- >User_SvrOK);

return 0;

}

//

// Copy the resolved information into the sockaddr_in structure

//

memset(&server,0,sizeof(server));

memcpy(&(server.sin_addr),hp- >h_addr,hp- >h_length);

server.sin_family = hp- >h_addrtype;

server.sin_port = htons(port);

conn_socket = socket(AF_INET,socket_type,0); /* 打开一个 socket */

if (conn_socket < 0 ) {

fprintf(stderr,"Client: Error Opening socket: Error %d\n",

WSAGetLastError());

pPar- >pPair- >IsProxy_ServerClosed=TRUE;

::SetEvent(pPar- >User_SvrOK);

return -1;

}

#ifdef _DEBUG

printf("Client connecting to: %s\n",hp- >h_name);

#endif

if (connect(conn_socket,(struct sockaddr*)&server,sizeof(server))

== SOCKET_ERROR) {

fprintf(stderr,"connect() failed: %d\n",WSAGetLastError());

pPar- >pPair- >IsProxy_ServerClosed=TRUE;

::SetEvent(pPar- >User_SvrOK);

return -1;

}

pPar- >pPair- >proxy_server=conn_socket;

pPar- >pPair- >IsProxy_ServerClosed=FALSE;

::SetEvent(pPar- >User_SvrOK);

// cook up a string to send

while(!pPar- >pPair- >IsProxy_ServerClosed &&!pPar- >pPair- >IsUser_ProxyClosed)

{

retval = recv(conn_socket,Buffer,sizeof (Buffer),0 );

if (retval == SOCKET_ERROR ) {

fprintf(stderr,"recv() failed: error %d\n",WSAGetLastError());

closesocket(conn_socket);

pPar- >pPair- >IsProxy_ServerClosed=TRUE;

break;

}

Len=retval;

if (retval == 0) {

printf("Server closed connection\n");

closesocket(conn_socket);

pPar- >pPair- >IsProxy_ServerClosed=TRUE;

break;

}

retval = send(pPar- >pPair- >user_proxy,Buffer,Len,0);

if (retval == SOCKET_ERROR) {

fprintf(stderr,"send() failed: error %d\n",WSAGetLastError());

closesocket(pPar- >pPair- >user_proxy);

pPar- >pPair- >IsUser_ProxyClosed=TRUE;

break;

}

#ifdef _DEBUG

Buffer[Len]=0;

printf("Received %d bytes, data [%s] from server\n",retval,Buffer);

#endif

}

if(pPar- >pPair- >IsProxy_ServerClosed==FALSE)

{

closesocket(pPar- >pPair- >proxy_server);

pPar- >pPair- >IsProxy_ServerClosed=TRUE;

}

if(pPar- >pPair- >IsUser_ProxyClosed==FALSE)

{closesocket(pPar- >pPair- >user_proxy);

pPar- >pPair- >IsUser_ProxyClosed=TRUE;

}

return 1;

}

int _tmain(int argc, TCHAR* argv[], TCHAR* envp[])

{

int nRetCode = 0;

// 初始化SOCKET

if (!AfxWinInit(::GetModuleHandle(NULL), NULL, ::GetCommandLine(), 0))

{

// 错误处理

cerr < < _T("Fatal Error: MFC initialization failed") < < endl;

nRetCode = 1;

}

else

{

// 主程序开始.

StartServer();

while(1)

if(getchar()=='q') break;

CloseServer();

}

return nRetCode;

}

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有