RealmSession.cs 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206
  1. using System;
  2. using System.Collections.Generic;
  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. namespace LoginClient
  10. {
  11. public class RealmSession: IDisposable
  12. {
  13. public int ID { get; set; }
  14. public IMessageChannel MessageChannel { get; set; }
  15. public RealmSession(int id, IMessageChannel messageChannel)
  16. {
  17. this.ID = id;
  18. this.MessageChannel = messageChannel;
  19. }
  20. public void Dispose()
  21. {
  22. this.MessageChannel.Dispose();
  23. }
  24. public async Task<SMSG_Password_Protect_Type> Handle_CMSG_AuthLogonPermit_Response()
  25. {
  26. var result = await this.MessageChannel.RecvMessage();
  27. ushort opcode = result.Item1;
  28. byte[] message = result.Item2;
  29. if (opcode != MessageOpcode.SMSG_PASSWORD_PROTECT_TYPE)
  30. {
  31. throw new LoginException(string.Format(
  32. "session: {0}, opcode: {1}", this.ID, opcode));
  33. }
  34. var smsgPasswordProtectType =
  35. ProtobufHelper.FromBytes<SMSG_Password_Protect_Type>(message);
  36. if (smsgPasswordProtectType.Code != 200)
  37. {
  38. throw new LoginException(string.Format(
  39. "session: {0}, SMSG_Lock_For_Safe_Time: {1}",
  40. this.ID, JsonHelper.ToString(smsgPasswordProtectType)));
  41. }
  42. return smsgPasswordProtectType;
  43. }
  44. public async Task<SMSG_Auth_Logon_Challenge_Response>
  45. Handle_SMSG_Auth_Logon_Challenge_Response()
  46. {
  47. var result = await this.MessageChannel.RecvMessage();
  48. ushort opcode = result.Item1;
  49. byte[] message = result.Item2;
  50. if (opcode != MessageOpcode.SMSG_AUTH_LOGON_CHALLENGE_RESPONSE)
  51. {
  52. Logger.Trace("opcode: {0}", opcode);
  53. throw new LoginException(string.Format(
  54. "session: {0}, opcode: {1}", this.ID, opcode));
  55. }
  56. var smsgAuthLogonChallengeResponse =
  57. ProtobufHelper.FromBytes<SMSG_Auth_Logon_Challenge_Response>(message);
  58. if (smsgAuthLogonChallengeResponse.ErrorCode != ErrorCode.REALM_AUTH_SUCCESS)
  59. {
  60. Logger.Trace("error code: {0}", smsgAuthLogonChallengeResponse.ErrorCode);
  61. throw new LoginException(
  62. string.Format("session: {0}, SMSG_Auth_Logon_Challenge_Response: {1}",
  63. this.ID, JsonHelper.ToString(smsgAuthLogonChallengeResponse)));
  64. }
  65. Logger.Debug("SMSG_Auth_Logon_Challenge_Response: \n{0}",
  66. JsonHelper.ToString(smsgAuthLogonChallengeResponse));
  67. return smsgAuthLogonChallengeResponse;
  68. }
  69. public async Task<SMSG_Auth_Logon_Proof_M2> Handle_SMSG_Auth_Logon_Proof_M2()
  70. {
  71. var result = await this.MessageChannel.RecvMessage();
  72. ushort opcode = result.Item1;
  73. byte[] message = result.Item2;
  74. if (opcode != MessageOpcode.SMSG_AUTH_LOGON_PROOF_M2)
  75. {
  76. throw new LoginException(string.Format(
  77. "session: {0}, error opcode: {1}", this.ID, opcode));
  78. }
  79. var smsgAuthLogonProofM2 = ProtobufHelper.FromBytes<SMSG_Auth_Logon_Proof_M2>(message);
  80. if (smsgAuthLogonProofM2.ErrorCode != ErrorCode.REALM_AUTH_SUCCESS)
  81. {
  82. throw new LoginException(string.Format(
  83. "session: {0}, SMSG_Auth_Logon_Proof_M2: {1}",
  84. this.ID, JsonHelper.ToString(smsgAuthLogonProofM2)));
  85. }
  86. return smsgAuthLogonProofM2;
  87. }
  88. public async Task<SMSG_Realm_List> Handle_SMSG_Realm_List()
  89. {
  90. var result = await this.MessageChannel.RecvMessage();
  91. ushort opcode = result.Item1;
  92. byte[] message = result.Item2;
  93. if (opcode != MessageOpcode.SMSG_REALM_LIST)
  94. {
  95. throw new LoginException(string.Format(
  96. "session: {0}, error opcode: {1}", this.ID, opcode));
  97. }
  98. var smsgRealmList = ProtobufHelper.FromBytes<SMSG_Realm_List>(message);
  99. return smsgRealmList;
  100. }
  101. public async Task<Tuple<string, ushort, SRP6Client>> Login(string account, string password)
  102. {
  103. byte[] passwordBytes = password.ToByteArray();
  104. MD5 md5 = MD5.Create();
  105. byte[] passwordMd5 = md5.ComputeHash(passwordBytes);
  106. byte[] passwordMd5Hex = passwordMd5.ToHex().ToLower().ToByteArray();
  107. // 发送帐号和密码MD5
  108. var cmsgAuthLogonPermit = new CMSG_Auth_Logon_Permit
  109. {
  110. Account = account.ToByteArray(),
  111. PasswordMd5 = passwordMd5Hex
  112. };
  113. Logger.Trace("session: {0}, account: {1}, password: {2}", this.ID,
  114. cmsgAuthLogonPermit.Account.ToStr(), cmsgAuthLogonPermit.PasswordMd5.ToHex());
  115. this.MessageChannel.SendMessage(MessageOpcode.CMSG_AUTH_LOGON_PERMIT, cmsgAuthLogonPermit);
  116. await this.Handle_CMSG_AuthLogonPermit_Response();
  117. // 这个消息已经没有作用,只用来保持原有的代码流程
  118. var cmsgAuthLogonChallenge = new CMSG_Auth_Logon_Challenge();
  119. this.MessageChannel.SendMessage(
  120. MessageOpcode.CMSG_AUTH_LOGON_CHALLENGE, cmsgAuthLogonChallenge);
  121. var smsgAuthLogonChallengeResponse =
  122. await this.Handle_SMSG_Auth_Logon_Challenge_Response();
  123. Logger.Trace("session: {0}, SMSG_Auth_Logon_Challenge_Response OK", this.ID);
  124. // 以下是SRP6处理过程
  125. var n = smsgAuthLogonChallengeResponse.N.ToUBigInteger();
  126. var g = smsgAuthLogonChallengeResponse.G.ToUBigInteger();
  127. var b = smsgAuthLogonChallengeResponse.B.ToUBigInteger();
  128. var salt = smsgAuthLogonChallengeResponse.S.ToUBigInteger();
  129. var srp6Client = new SRP6Client(
  130. new SHA1Managed(), n, g, b, salt, account.ToByteArray(), passwordMd5Hex);
  131. Logger.Debug("s: {0}\nN: {1}\nG: {2}\nB: {3}\nA: {4}\nS: {5}\nK: {6}\nm: {7}\na: {8}",
  132. srp6Client.Salt.ToUBigIntegerArray().ToHex(),
  133. srp6Client.N.ToUBigIntegerArray().ToHex(),
  134. srp6Client.G.ToUBigIntegerArray().ToHex(),
  135. srp6Client.B.ToUBigIntegerArray().ToHex(),
  136. srp6Client.A.ToUBigIntegerArray().ToHex(),
  137. srp6Client.S.ToUBigIntegerArray().ToHex(),
  138. srp6Client.K.ToUBigIntegerArray().ToHex(),
  139. srp6Client.M.ToUBigIntegerArray().ToHex(),
  140. srp6Client.SmallA.ToUBigIntegerArray().ToHex());
  141. var cmsgAuthLogonProof = new CMSG_Auth_Logon_Proof
  142. {
  143. A = srp6Client.A.ToUBigIntegerArray(),
  144. M = srp6Client.M.ToUBigIntegerArray()
  145. };
  146. this.MessageChannel.SendMessage(MessageOpcode.CMSG_AUTH_LOGON_PROOF, cmsgAuthLogonProof);
  147. await this.Handle_SMSG_Auth_Logon_Proof_M2();
  148. Logger.Trace("session: {0}, SMSG_Auth_Logon_Proof_M2 OK", this.ID);
  149. // 请求realm list
  150. var cmsgRealmList = new CMSG_Realm_List();
  151. this.MessageChannel.SendMessage(MessageOpcode.CMSG_REALM_LIST, cmsgRealmList);
  152. var smsgRealmList = await this.Handle_SMSG_Realm_List();
  153. Logger.Trace("session: {0}, SMSG_Realm_List OK", this.ID);
  154. string address = smsgRealmList.GateList[0].Address.ToStr();
  155. string[] split = address.Split(new[] { ':' });
  156. if (split.Length != 2)
  157. {
  158. throw new LoginException(
  159. string.Format("session: {0}, gate address error, address: {1}",
  160. this.ID, address));
  161. }
  162. string gateIP = split[0];
  163. ushort gatePort = UInt16.Parse(split[1]);
  164. return Tuple.Create(gateIP, gatePort, srp6Client);
  165. }
  166. }
  167. }