RealmSession.cs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. using System;
  2. using System.Linq;
  3. using System.Net.Sockets;
  4. using System.Numerics;
  5. using System.Security.Cryptography;
  6. using System.Threading.Tasks;
  7. using Helper;
  8. using Log;
  9. using Robot.Protos;
  10. namespace Robot
  11. {
  12. public class RealmSession: IDisposable
  13. {
  14. private readonly TcpClient tcpClient = new TcpClient();
  15. private NetworkStream networkStream;
  16. private readonly RealmInfo realmInfo = new RealmInfo();
  17. public void Dispose()
  18. {
  19. this.tcpClient.Close();
  20. this.networkStream.Dispose();
  21. }
  22. public async void SendMessage<T>(ushort opcode, T message)
  23. {
  24. byte[] protoBytes = ProtobufHelper.ToBytes(message);
  25. var neworkBytes = new byte[sizeof (int) + sizeof (ushort) + protoBytes.Length];
  26. int totalSize = sizeof (ushort) + protoBytes.Length;
  27. var totalSizeBytes = BitConverter.GetBytes(totalSize);
  28. totalSizeBytes.CopyTo(neworkBytes, 0);
  29. var opcodeBytes = BitConverter.GetBytes(opcode);
  30. opcodeBytes.CopyTo(neworkBytes, sizeof (int));
  31. protoBytes.CopyTo(neworkBytes, sizeof (int) + sizeof (ushort));
  32. await this.networkStream.WriteAsync(neworkBytes, 0, neworkBytes.Length);
  33. }
  34. public async Task<SMSG_Password_Protect_Type> Handle_CMSG_AuthLogonPermit_Response()
  35. {
  36. var result = await this.RecvMessage();
  37. ushort opcode = result.Item1;
  38. byte[] message = result.Item2;
  39. if (opcode != MessageOpcode.SMSG_PASSWORD_PROTECT_TYPE)
  40. {
  41. Logger.Trace("opcode: {0}", opcode);
  42. throw new RealmException(string.Format("error opcode: {0}", opcode));
  43. }
  44. var smsgPasswordProtectType =
  45. ProtobufHelper.FromBytes<SMSG_Password_Protect_Type>(message);
  46. Logger.Trace("message: {0}", JsonHelper.ToString(smsgPasswordProtectType));
  47. if (smsgPasswordProtectType.Code != 200)
  48. {
  49. throw new RealmException(string.Format(
  50. "SMSG_Lock_For_Safe_Time: {0}",
  51. JsonHelper.ToString(smsgPasswordProtectType)));
  52. }
  53. return smsgPasswordProtectType;
  54. }
  55. public async Task<SMSG_Auth_Logon_Challenge_Response>
  56. Handle_SMSG_Auth_Logon_Challenge_Response()
  57. {
  58. var result = await this.RecvMessage();
  59. ushort opcode = result.Item1;
  60. byte[] message = result.Item2;
  61. if (opcode != MessageOpcode.SMSG_AUTH_LOGON_CHALLENGE_RESPONSE)
  62. {
  63. Logger.Trace("opcode: {0}", opcode);
  64. }
  65. var smsgAuthLogonChallengeResponse =
  66. ProtobufHelper.FromBytes<SMSG_Auth_Logon_Challenge_Response>(message);
  67. if (smsgAuthLogonChallengeResponse.ErrorCode != ErrorCode.REALM_AUTH_SUCCESS)
  68. {
  69. Logger.Trace("error code: {0}", smsgAuthLogonChallengeResponse.ErrorCode);
  70. throw new RealmException(
  71. string.Format("SMSG_Auth_Logon_Challenge_Response ErrorCode: {0}",
  72. JsonHelper.ToString(smsgAuthLogonChallengeResponse)));
  73. }
  74. return smsgAuthLogonChallengeResponse;
  75. }
  76. public async Task<Tuple<ushort, byte[]>> RecvMessage()
  77. {
  78. int totalReadSize = 0;
  79. int needReadSize = sizeof (int);
  80. var packetBytes = new byte[needReadSize];
  81. while (totalReadSize != needReadSize)
  82. {
  83. int readSize = await this.networkStream.ReadAsync(
  84. packetBytes, totalReadSize, packetBytes.Length);
  85. if (readSize == 0)
  86. {
  87. throw new RealmException("connection closed!");
  88. }
  89. totalReadSize += readSize;
  90. }
  91. int packetSize = BitConverter.ToInt32(packetBytes, 0);
  92. Logger.Debug("packet size: {0}", packetSize);
  93. // 读opcode和message
  94. totalReadSize = 0;
  95. needReadSize = packetSize;
  96. var contentBytes = new byte[needReadSize];
  97. while (totalReadSize != needReadSize)
  98. {
  99. int readSize = await this.networkStream.ReadAsync(
  100. contentBytes, totalReadSize, contentBytes.Length);
  101. if (readSize == 0)
  102. {
  103. throw new RealmException("connection closed!");
  104. }
  105. totalReadSize += readSize;
  106. }
  107. ushort opcode = BitConverter.ToUInt16(contentBytes, 0);
  108. Logger.Debug("opcode: {0}", opcode);
  109. var messageBytes = new byte[needReadSize - sizeof (ushort)];
  110. Array.Copy(contentBytes, sizeof (ushort), messageBytes, 0, messageBytes.Length);
  111. return new Tuple<ushort, byte[]>(opcode, messageBytes);
  112. }
  113. public async Task ConnectAsync(string hostName, ushort port)
  114. {
  115. await this.tcpClient.ConnectAsync(hostName, port);
  116. this.networkStream = this.tcpClient.GetStream();
  117. }
  118. public async void Login(string account, string password)
  119. {
  120. byte[] passwordBytes = password.ToByteArray();
  121. MD5 md5 = MD5.Create();
  122. byte[] passwordMd5 = md5.ComputeHash(passwordBytes);
  123. // 发送帐号和密码MD5
  124. var cmsgAuthLogonPermit = new CMSG_Auth_Logon_Permit
  125. {
  126. Account = account.ToByteArray(),
  127. PasswordMd5 = passwordMd5.ToHex().ToLower().ToByteArray()
  128. };
  129. Logger.Trace("account: {0}, password: {1}",
  130. cmsgAuthLogonPermit.Account, cmsgAuthLogonPermit.PasswordMd5.ToStr());
  131. this.SendMessage(MessageOpcode.CMSG_AUTH_LOGON_PERMIT, cmsgAuthLogonPermit);
  132. await this.Handle_CMSG_AuthLogonPermit_Response();
  133. // 这个消息已经没有作用,只用来保持原有的代码流程
  134. var cmsgAuthLogonChallenge = new CMSG_Auth_Logon_Challenge();
  135. this.SendMessage(MessageOpcode.CMSG_AUTH_LOGON_CHALLENGE, cmsgAuthLogonChallenge);
  136. var smsgAuthLogonChallengeResponse =
  137. await this.Handle_SMSG_Auth_Logon_Challenge_Response();
  138. // 以下是SRP6处理过程
  139. var n = smsgAuthLogonChallengeResponse.N.ToUnsignedBigInteger();
  140. var g = smsgAuthLogonChallengeResponse.G.ToUnsignedBigInteger();
  141. var s = smsgAuthLogonChallengeResponse.S.ToUnsignedBigInteger();
  142. var b = smsgAuthLogonChallengeResponse.B.ToUnsignedBigInteger();
  143. string identity = account + ":" + password;
  144. var srp6Client = new SRP6Client(new SHA1Managed(), n, g, b, s, identity, password);
  145. Logger.Debug("N: {0}\nG: {1}\ns: {2}\nB: {3}\nA: {4}\nS: {5}\nK: {6}\nm: {7}",
  146. srp6Client.N.ToTrimByteArray().ToHex(), srp6Client.G.ToTrimByteArray().ToHex(),
  147. srp6Client.S.ToTrimByteArray().ToHex(), srp6Client.B.ToTrimByteArray().ToHex(),
  148. srp6Client.A.ToTrimByteArray().ToHex(), srp6Client.S.ToTrimByteArray().ToHex(),
  149. srp6Client.K.ToTrimByteArray().ToHex(), srp6Client.M.ToHex());
  150. var cmsgAuthLogonProof = new CMSG_Auth_Logon_Proof
  151. {
  152. A = srp6Client.A.ToTrimByteArray(),
  153. M = srp6Client.M
  154. };
  155. this.SendMessage(MessageOpcode.CMSG_AUTH_LOGON_PROOF, cmsgAuthLogonProof);
  156. }
  157. }
  158. }