OCPPWSServer.cs 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. 
  2. using EVCB_OCPP.Domain;
  3. using EVCB_OCPP.WSServer;
  4. using EVCB_OCPP.WSServer.Helper;
  5. using NLog;
  6. using OCPPPackage.Profiles;
  7. using SuperWebSocket;
  8. using SuperWebSocket.SubProtocol;
  9. using System;
  10. using System.Collections.Generic;
  11. using System.Configuration;
  12. using System.Linq;
  13. using System.Net;
  14. using System.Net.Security;
  15. using System.Security.Cryptography.X509Certificates;
  16. using System.Text;
  17. namespace OCPPServer.Protocol
  18. {
  19. public class OCPPWSServer : WebSocketServer<ClientData>
  20. {
  21. static private ILogger logger = NLog.LogManager.GetCurrentClassLogger();
  22. /// <summary>
  23. /// 可允許連線Clinet數
  24. /// </summary>
  25. public int connectNum { get; set; }
  26. /// <summary>
  27. /// 是否限制連線Clinet數
  28. /// </summary>
  29. public bool beConnectLimit { get; set; }
  30. /// <summary>
  31. /// Initializes a new instance of the <see cref="WebSocketServer"/> class.
  32. /// </summary>
  33. /// <param name="subProtocols">The sub protocols.</param>
  34. public OCPPWSServer(IEnumerable<ISubProtocol<ClientData>> subProtocols)
  35. : base(subProtocols)
  36. {
  37. }
  38. /// <summary>
  39. /// Initializes a new instance of the <see cref="WebSocketServer"/> class.
  40. /// </summary>
  41. /// <param name="subProtocol">The sub protocol.</param>
  42. public OCPPWSServer(ISubProtocol<ClientData> subProtocol)
  43. : base(subProtocol)
  44. {
  45. }
  46. /// <summary>
  47. /// Initializes a new instance of the <see cref="WebSocketServer"/> class.
  48. /// </summary>
  49. public OCPPWSServer()
  50. : base(new List<ISubProtocol<ClientData>>())
  51. {
  52. }
  53. protected override bool ValidateClientCertificate(ClientData session, object sender, X509Certificate certificate, X509Chain chain, SslPolicyErrors sslPolicyErrors)
  54. {
  55. // Console.WriteLine(string.Format("{0} :{1}", session.ChargeBoxId + " ValidateClientCertificate", sslPolicyErrors));
  56. return true;
  57. // return base.ValidateClientCertificate(session, sender, certificate, chain, sslPolicyErrors);
  58. }
  59. protected override bool ValidateHandshake(ClientData session, string origin)
  60. {
  61. int securityProfile = GlobalConfig.SecurityProfileLevel;
  62. string authorizationKey = GlobalConfig.SecurityPassword;
  63. bool authorizated = false;
  64. session.ISOCPP20 = session.SecWebSocketProtocol.ToLower().Contains("ocpp2.0");
  65. if (string.IsNullOrEmpty(session.Path))
  66. {
  67. logger.Warn("===========================================");
  68. logger.Warn("session.Path EMPTY");
  69. logger.Warn("===========================================");
  70. }
  71. string[] words = session.Path.Split('/');
  72. session.ChargeBoxId = words.Last();
  73. if (ConfigurationManager.AppSettings["MaintainMode"] == "1")
  74. {
  75. session.ChargeBoxId = session.ChargeBoxId + "_2";
  76. }
  77. logger.Info(string.Format("ValidateHandshake: {0}", session.Path));
  78. if (session.ISOCPP20)
  79. {
  80. // 1.6 server only support change server function
  81. securityProfile = 0;
  82. }
  83. switch (securityProfile)
  84. {
  85. case 0:
  86. {
  87. authorizated = true;
  88. }
  89. break;
  90. case 1:
  91. {
  92. if(session.UriScheme == "ws")
  93. {
  94. string client_authorization = string.Empty;
  95. if (session.Items.ContainsKey("Authorization") || session.Items.ContainsKey("authorization"))
  96. {
  97. client_authorization = session.Items.ContainsKey("Authorization") ? session.Items["Authorization"].ToString().Replace("Basic ", "") : session.Items["authorization"].ToString().Replace("Basic ", "");
  98. }
  99. if (ValidateSecurity(session.ChargeBoxId, authorizationKey, client_authorization))
  100. {
  101. authorizated = true;
  102. }
  103. else
  104. {
  105. logger.Warn(string.Format(" {0} ValidateSecurity Fail client_authorization:[{1}]:", session.ChargeBoxId, client_authorization));
  106. }
  107. }
  108. }
  109. break;
  110. case 2:
  111. {
  112. if (session.UriScheme == "wss")
  113. {
  114. string client_authorization = string.Empty;
  115. if (session.Items.ContainsKey("Authorization") || session.Items.ContainsKey("authorization"))
  116. {
  117. client_authorization = session.Items.ContainsKey("Authorization") ? session.Items["Authorization"].ToString().Replace("Basic ", "") : session.Items["authorization"].ToString().Replace("Basic ", "");
  118. }
  119. if (ValidateSecurity(session.ChargeBoxId, authorizationKey, client_authorization))
  120. {
  121. authorizated = true;
  122. }
  123. else
  124. {
  125. logger.Warn(string.Format(" {0} ValidateSecurity Fail client_authorization:[{1}]:", session.ChargeBoxId, client_authorization));
  126. }
  127. }
  128. }
  129. break;
  130. case 3:
  131. {
  132. authorizated = false;
  133. }
  134. break;
  135. default:
  136. break;
  137. }
  138. if(!authorizated)
  139. {
  140. StringBuilder responseBuilder = new StringBuilder();
  141. responseBuilder.AppendFormatWithCrCf(@"HTTP/{0} {1} {2}", "1.1",
  142. (int)HttpStatusCode.Unauthorized, @"Unauthorized");
  143. responseBuilder.AppendWithCrCf();
  144. string sb = responseBuilder.ToString();
  145. byte[] data = Encoding.UTF8.GetBytes(sb);
  146. ((IWebSocketSession)session).SendRawData(data, 0, data.Length);
  147. logger.Info(sb);
  148. return false;
  149. }
  150. return true;
  151. }
  152. private bool ValidateSecurity(string chargeBoxId, string authorizationKey, string client_authorization)
  153. {
  154. bool result = false;
  155. if (string.IsNullOrEmpty(client_authorization)) return result;
  156. string server_authorization = EncodeHelper.EncodeBase64(string.Format("{0}:{1}", chargeBoxId, authorizationKey));
  157. if (client_authorization == server_authorization)
  158. {
  159. result = true;
  160. }
  161. return result;
  162. }
  163. }
  164. }