myicq-1.0a1服务器代码分析(一)

本文对myicq-1.0a1服务器端代码进行分析,着重探讨了UDP服务的实现。包括数据缓冲包的类管理,定义抽象类及派生类,进行数据加密解密处理;socket服务代码编写,定义多个类及相关接口。同时指出代码存在执行效率不高、可能溢出、模块化低等问题,并给出插件机制的改进建议。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

myicq-1.0a1服务器代码分析(一)

顾剑辉(Solarsoft)

myicq代码的公布已经有一段时间了,听说作者张勇已经不再公开的服务器端的代码了,不尤觉得可惜,拜读他的作品已经有一段时间了,今天来发表一下自己的意见。

我现在就从整体的构架来谈谈他服务器端的代码,服务器代码可分成数据库、upd服务、服务器群组、插件四块。

我这里对upd服务的实现进行一些讨论,

一、数据缓冲包的类管理

先定义了抽象类OutPacketInPacket,代码如下:

class OutPacket {

public:

       virtual OutPacket &operator <<(uint8 b) = 0;

       virtual OutPacket &operator <<(uint16 w) = 0;

       virtual OutPacket &operator <<(uint32 dw) = 0;

       virtual OutPacket &operator <<(ICQ_STR &str) = 0;

};

 

 

class InPacket {

public:

       virtual InPacket &operator >>(uint8 &b) = 0;

       virtual InPacket &operator >>(uint16 &w) = 0;

       virtual InPacket &operator >>(uint32 &dw) = 0;

       virtual InPacket &operator >>(ICQ_STR &str) = 0;

};

在实现其派生类IcqOutPacketIcqInPacket,并定义数据缓冲区uint8 data[UDP_PACKET_SIZE],大小为1024,主要是处理数据的输入与输出。

再派生其类UdpOutPacketUdpInPacket,此时进行了DES的数据加密与解密的处理,并加入数据头,定义如下:

struct UDP_CLI_HDR {

       uint16 ver;

       uint32 reserved;

       uint32 uin;

       uint32 sid;

       uint16 cmd;

       uint16 seq;

       uint16 cc;              // check code

};

 

struct UDP_SRV_HDR {

       uint16 ver;

       uint32 reserved;

       uint32 uin;

       uint32 sid;

       uint16 cmd;

       uint16 seq;

       uint16 ackseq;

};

此时缓冲区的包类已经结束。

二、socket服务代码的编写

先定义2个类RefObject Session,其中RefObject为记数类,两代码如下:

class RefObject {

public:

       RefObject() {

              refCount = 1;

       }

 

       int addRef() {

              return ++refCount;

       }

       int release() {

              register int ret = --refCount;

              if (!ret)

                     delete this;

              return ret;

       }

 

protected:

       virtual ~RefObject() {}

 

       int refCount;

};

class Session {

public:

       uint32 uin;

       uint32 status;

       uint32 ip;

       uint16 msgport;

       uint32 realip;

};

现在我们来看一下udp服务类的定义

class UdpSession : public RefObject, public Session {

public:

       UdpSession(UdpInPacket &in, uint32 ip, uint16 port);

       ~UdpSession();

 

       UdpOutPacket *createPacket(uint16 cmd, uint16 ackseq = 0);

       void sendPacket(UdpOutPacket *p);

 

       void sendOnline(Session *s, ICQ_STR &domain);

       void sendOffline(uint32 uin, ICQ_STR &domain);

       void sendStatusChanged(Session *s, ICQ_STR &domain);

       void sendMessage(uint8 type, QID &src, uint32 when, ICQ_STR &text);

       void sendGroupTypes();

       void updateContactReply(uint16 seq, uint8 *data, int n, Server *server = NULL);

       void searchRandomReply(uint16 seq, uint8 *data, int n);

       void searchUINReply(uint16 seq, uint8 *data, int n);

 

       void dead();

       void logout();

 

       void onlineNotify();

       void offlineNotify();

       void statusNotify(uint32 newStatus);

 

       static bool init();

       static void destroy();

       static bool onReceive();

       static void checkSendQueue();

       static void checkKeepAlive();

 

       static void sendMessage(uint8 type, QID &dst, QID &src, uint32 when, ICQ_STR &text);

       static void addFriend(uint16 seq, QID &dst, uint32 src, Server *server = NULL);

       static void addFriendAuth(uint16 seq, QID &dst, uint32 src, Server *server, uint8 auth, bool sendAuthMsg = true);

       static void updateContact(uint16 seq, uint32 dst, uint32 src, Server *server = NULL);

       static void searchUIN(uint16 seq, uint32 dst, uint32 src, Server *server = NULL);

 

       uint8 auth;

       uint8 face;

       char nickname[MAX_NICK + 1];

       char province[MAX_PROVINCE + 1];

 

       uint16 tcpver;

       uint16 port;

       uint32 oldMsgID, lastMsgID;

       char subkey[128];

 

       ListHead uinItem;

       ListHead ipportItem;

       ListHead listItem;

 

       static int sock;

       static uint32 sessionCount;

 

private:

       void createPacket(UdpOutPacket &out, uint16 cmd, uint16 seq, uint16 ackseq);

       void sendDirect(UdpOutPacket *p);

       void sendAckPacket(uint16 seq);

       bool setWindow(uint16 seq);

       void notify(DB_CALLBACK cb1, DB_CALLBACK cb2);

 

       void onAck(uint16 seq);

       void onKeepAlive(UdpInPacket &in);

       void onNewUIN(UdpInPacket &in);

       void onGetContactList(UdpInPacket &in);

       void onGetRemoteContactList(UdpInPacket &in);

       void onLogin(UdpInPacket &in);

       void onLogout(UdpInPacket &in);

       void onChangeStatus(UdpInPacket &in);

       void onUpdateContact(UdpInPacket &in);

       void onModifyUser(UdpInPacket &in);

       void onUpdateUser(UdpInPacket &in);

       void onSendMessage(UdpInPacket &in);

       void onSearchRandom(UdpInPacket &in);

       void onSearchCustom(UdpInPacket &in);

       void onAddFriend(UdpInPacket &in);

       void onDelFriend(UdpInPacket &in);

       void onSendBCMsg(UdpInPacket &in);

       void onGetServerList(UdpInPacket &in);

       void onGetGroupList(UdpInPacket &in);

       void onSearchGroup(UdpInPacket &in);

 

       void onCreateGroup(UdpInPacket &in);

       void onEnterGroup(UdpInPacket &in);

       void onExitGroup(UdpInPacket &in);

       void onGroupStart(UdpInPacket &in);

       void onGroupMessage(UdpInPacket &in);

       void onGroupCmd(UdpInPacket &in);

 

       bool onPacketReceived(UdpInPacket &in);

 

       uint16 udpver;

       uint32 sid;

       uint16 sendSeq;

       uint16 recvSeq;

       uint32 window;

       time_t expire;

 

       uint8 isDead : 1;

 

       IcqGroup *group;

 

       ListHead sendQueue;

 

       DECLARE_SLAB(UdpSession)

};

都是一些与客户端通信的接口。

我的评论与建议:这样的数据缓冲区的设计,代码执行效率不是很高,而且可能造成服务器缓冲区溢出的错误,应该设计时加入一些防范措施。

UdpSession的设计不太合理,模块化不高,代码零乱。我到认为这里到可以用插件机制,也就是一个插件来管理部分的通信消息命令。这样可提高程序的可读性,还便于今后通信消息命令的扩展。这点可以学学入侵检测系统Snort的规则处理模块与插件的管理。

 

 

MyICQ 0.8 alpha1测试版 ==================================================== --------------- 1. MyICQ是什么? --------------- MyICQ套公开源代码的即时通讯软件,包括服务器端和客户端,可以用于互联网或局域网中。可以运行在Windows或Linux(KDE/Qt)操作系统上,这是Windows版。目前客户端程序的界面完全模仿腾讯的QQ(如果Tencent告我的话,我会马上改的:-)。 总之,如果你崇尚自由,对QQ的越来越多的广告骚扰感到深恶痛绝的话,MyICQ绝对是你很好的选择。 -------- 2. 版本 -------- 这是MyICQ 0.8版本的alpha1 测试版,基本上还没有经过什么严格测试。 -------- 3. 版权 -------- 完全遵循GPL协议2.0或以后协议版本。 --------------- 4. 基本功能特点 --------------- 1) 收发(离线)消息(如果客户端之间能直接通讯,则通过UDP协议发送,否则通过服务器中转) 2) 添加/删除好友(可以设置身份验证) 3) 服务器端存储好友列表 4) 在客户端存储好友资料和聊天记录 5) 客户端与服务器端用DES的密钥加密方式通讯 6) 支持代理服务器(SOCKS5/HTTP) 7) 向在线的组好友发送消息 8) 系统管理员(MyICQ号 < 1000)可以发送系统广播消息 9) 皮肤系统 10)完全基于插件,使扩展更容易。目前提供的插件有: a. 收发文件 b. 二人世界(可以实时语音聊天) c. 闹钟提醒 ------------------ 5. 分发包里有什么? ------------------ 分发包里有三个目录: myicq/ 客户端程序 myicqd/ 服务器端程序 myicqhttp/ HTTP的转换程序。MyICQ支持HTTP代理,但需要在MyICQ服务器端运行myicqhttp程序 -------- 6. 安装 -------- 客户端程序不写Windows注册表,无需安装。双击MyICQ.exe运行。 服务器端由于内部采用MySQL数据库,所以安装稍微麻烦些: 1) 到http://www.mysql.com下载MySQL 2) 打开个DOS命令窗口,在MySQL中建立个数据库,并添加个用户: C:\mysql\bin> mysql -uroot -p ******** (输入root密码,如果还没有设置,直接输入回车即可) mysql> GRANT ALL ON myicq.* TO myicq@localhost IDENTIFIED BY 'myicq'; mysql> CREATE DATABASE myicq; mysql> quit 3) 创建表格: C:\mysql\bin> mysql -umyicq -Dmyicq -p < [myicqd目录]\myicq.sql password: myicq 注意: [myicqd目录]代表myicqd所在的目录 4) 运行myicqd: 双击myicqd.exe即可运行。如果没有错误,应该显示"MyICQ server is now started". 接下来就可以从客户端注册新用户了。 5) 你可能想要添加个系统用户,以发送系统广播消息: C:\mysql\bin> mysql -umyicq -Dmyicq -p password: myicq mysql> INSERT INTO basic_tbl (uin, passwd) VALUES(100, password('yourpassword')); mysql> INSERT INTO ext_tbl (uin) VALUES(100); 注意: yourpassword代表系统用户的密码 然后选择客户端程序的注册向导,取回100这个号码。完成后,你会发现在主菜单中多了"广播消息"项。此后,你可以选择个人设定来设定你的个人信息(比如,不允许任何人加我为好友) ------------- 7. 编译源代码 ------------- 如果你是个程序员,那么定要编译MyICQ的源代码:-) 1) 到http://www.mysql.com下载MySQL 2) 由于MyICQ在存储好友资料和聊天记录时,使用Berkeley DB库,所以先到 http://www.sleepycat.com下载 3) 编译Berkeley DB 4) 在VC的Tools->Options->Directories中设置好MySQL和BerkeleyDB的include和library路径 5) 在VC中打开myicq-win32/src/win32/myicq.dsw项目文件 6) 编译 --------- 8. TODO --------- 1) POP3邮件自动监测(插件) 2) 文件共享(插件) 3) 些小游戏(插件,如俄罗斯方块) 4) 发送好友/URL消息 5) 更完善的皮肤系统 6) 服务器端改为Windows服务程序 7) 服务器端的性能和稳定性 ----------- 9. 共同开发 ----------- MyICQ是我利用上学和上班的空闲时间编写的,发展到现在已经是个比较大的工程了,以我个人的力量继续编写下去将非常困难。希望广大的程序员高手能够参与到这个项目中来,为自由软件贡献份力量。 我在sourceforge.net上申请了CVS服务空间,项目名是myicq-free。有意加入者请与我联系。 ----------- 10. 联系方式 ----------- 如果你在使用MyICQ的过程中,遇到了什么BUG,或是有什么新的想法和建议,或是喜欢这个程序,我将非常高兴收到你的来信。 Email: z-yong163@163.com 张勇 Linux用户请不要着急,等在Windows下测试稳定后,马上就会出Linux版的! 天才的人是流星,注定要燃烧自己以照亮他们的世纪 ----- 拿破仑波拿巴 辆法拉利小车只是物质享受,做人最重要的是真诚 ----- John Carmack 什么叫专业,这就... ----- 张勇:-)
MyICQ 0.8 alpha1测试版 ==================================================== --------------- 1. MyICQ是什么? --------------- MyICQ套公开源代码的即时通讯软件,包括服务器端和客户端,可以用于互联网或局域网中。可以运行在Windows或Linux(KDE/Qt)操作系统上,这是Windows版。目前客户端程序的界面完全模仿腾讯的QQ(如果Tencent告我的话,我会马上改的:-)。 总之,如果你崇尚自由,对QQ的越来越多的广告骚扰感到深恶痛绝的话,MyICQ绝对是你很好的选择。 -------- 2. 版本 -------- 这是MyICQ 0.8版本的alpha1 测试版,基本上还没有经过什么严格测试。 -------- 3. 版权 -------- 完全遵循GPL协议2.0或以后协议版本。 --------------- 4. 基本功能特点 --------------- 1) 收发(离线)消息(如果客户端之间能直接通讯,则通过UDP协议发送,否则通过服务器中转) 2) 添加/删除好友(可以设置身份验证) 3) 服务器端存储好友列表 4) 在客户端存储好友资料和聊天记录 5) 客户端与服务器端用DES的密钥加密方式通讯 6) 支持代理服务器(SOCKS5/HTTP) 7) 向在线的组好友发送消息 8) 系统管理员(MyICQ号 < 1000)可以发送系统广播消息 9) 皮肤系统 10)完全基于插件,使扩展更容易。目前提供的插件有: a. 收发文件 b. 二人世界(可以实时语音聊天) c. 闹钟提醒 ------------------ 5. 分发包里有什么? ------------------ 分发包里有三个目录: myicq/ 客户端程序 myicqd/ 服务器端程序 myicqhttp/ HTTP的转换程序。MyICQ支持HTTP代理,但需要在MyICQ服务器端运行myicqhttp程序 -------- 6. 安装 -------- 客户端程序不写Windows注册表,无需安装。双击MyICQ.exe运行。 服务器端由于内部采用MySQL数据库,所以安装稍微麻烦些: 1) 到http://www.mysql.com下载MySQL 2) 打开个DOS命令窗口,在MySQL中建立个数据库,并添加个用户: C:\mysql\bin> mysql -uroot -p ******** (输入root密码,如果还没有设置,直接输入回车即可) mysql> GRANT ALL ON myicq.* TO myicq@localhost IDENTIFIED BY 'myicq'; mysql> CREATE DATABASE myicq; mysql> quit 3) 创建表格: C:\mysql\bin> mysql -umyicq -Dmyicq -p < [myicqd目录]\myicq.sql password: myicq 注意: [myicqd目录]代表myicqd所在的目录 4) 运行myicqd: 双击myicqd.exe即可运行。如果没有错误,应该显示"MyICQ server is now started". 接下来就可以从客户端注册新用户了。 5) 你可能想要添加个系统用户,以发送系统广播消息: C:\mysql\bin> mysql -umyicq -Dmyicq -p password: myicq mysql> INSERT INTO basic_tbl (uin, passwd) VALUES(100, password('yourpassword')); mysql> INSERT INTO ext_tbl (uin) VALUES(100); 注意: yourpassword代表系统用户的密码 然后选择客户端程序的注册向导,取回100这个号码。完成后,你会发现在主菜单中多了"广播消息"项。此后,你可以选择个人设定来设定你的个人信息(比如,不允许任何人加我为好友) ------------- 7. 编译源代码 ------------- 如果你是个程序员,那么定要编译MyICQ的源代码:-) 1) 到http://www.mysql.com下载MySQL 2) 由于MyICQ在存储好友资料和聊天记录时,使用Berkeley DB库,所以先到 http://www.s
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值