/src/Iscsi/ChapAuthenticator.cs

# · C# · 136 lines · 96 code · 19 blank · 21 comment · 7 complexity · dd4b1d044ba661c926e782a4d787328d MD5 · raw file

  1. //
  2. // Copyright (c) 2008-2011, Kenneth Bell
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a
  5. // copy of this software and associated documentation files (the "Software"),
  6. // to deal in the Software without restriction, including without limitation
  7. // the rights to use, copy, modify, merge, publish, distribute, sublicense,
  8. // and/or sell copies of the Software, and to permit persons to whom the
  9. // Software is furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
  20. // DEALINGS IN THE SOFTWARE.
  21. //
  22. namespace DiscUtils.Iscsi
  23. {
  24. using System;
  25. using System.Globalization;
  26. using System.Security.Cryptography;
  27. using System.Text;
  28. internal class ChapAuthenticator : Authenticator
  29. {
  30. private State _state;
  31. private string _name;
  32. private string _password;
  33. private int _algorithm;
  34. private byte _identifier;
  35. private byte[] _challenge;
  36. public ChapAuthenticator(string name, string password)
  37. {
  38. _name = name;
  39. _password = password;
  40. _state = State.SendAlgorithm;
  41. }
  42. private enum State
  43. {
  44. SendAlgorithm,
  45. ReceiveChallenge,
  46. SendResponse,
  47. Finished
  48. }
  49. public override string Identifier
  50. {
  51. get { return "CHAP"; }
  52. }
  53. public override bool GetParameters(TextBuffer textBuffer)
  54. {
  55. switch (_state)
  56. {
  57. case State.SendAlgorithm:
  58. textBuffer.Add("CHAP_A", "5");
  59. _state = State.ReceiveChallenge;
  60. return false;
  61. case State.SendResponse:
  62. textBuffer.Add("CHAP_N", _name);
  63. textBuffer.Add("CHAP_R", CalcResponse());
  64. _state = State.Finished;
  65. return true;
  66. default:
  67. throw new InvalidOperationException("Unknown authentication state: " + _state);
  68. }
  69. }
  70. public override void SetParameters(TextBuffer textBuffer)
  71. {
  72. switch (_state)
  73. {
  74. case State.ReceiveChallenge:
  75. _algorithm = int.Parse(textBuffer["CHAP_A"], CultureInfo.InvariantCulture);
  76. _identifier = byte.Parse(textBuffer["CHAP_I"], CultureInfo.InvariantCulture);
  77. _challenge = ParseByteString(textBuffer["CHAP_C"]);
  78. _state = State.SendResponse;
  79. if (_algorithm != 0x5)
  80. {
  81. throw new LoginException("Unexpected CHAP authentication algorithm: " + _algorithm);
  82. }
  83. return;
  84. default:
  85. throw new InvalidOperationException("Unknown authentication state: " + _state);
  86. }
  87. }
  88. private static byte[] ParseByteString(string p)
  89. {
  90. if (!p.StartsWith("0x", StringComparison.OrdinalIgnoreCase))
  91. {
  92. throw new InvalidProtocolException("Invalid value in CHAP exchange");
  93. }
  94. byte[] data = new byte[(p.Length - 2) / 2];
  95. for (int i = 0; i < data.Length; ++i)
  96. {
  97. data[i] = byte.Parse(p.Substring(2 + (i * 2), 2), NumberStyles.HexNumber, CultureInfo.InvariantCulture);
  98. }
  99. return data;
  100. }
  101. private string CalcResponse()
  102. {
  103. MD5 md5 = MD5CryptoServiceProvider.Create();
  104. byte[] toHash = new byte[1 + _password.Length + _challenge.Length];
  105. toHash[0] = _identifier;
  106. Encoding.ASCII.GetBytes(_password, 0, _password.Length, toHash, 1);
  107. Array.Copy(_challenge, 0, toHash, _password.Length + 1, _challenge.Length);
  108. byte[] hash = md5.ComputeHash(toHash);
  109. StringBuilder result = new StringBuilder("0x");
  110. for (int i = 0; i < hash.Length; ++i)
  111. {
  112. result.Append(hash[i].ToString("x2", CultureInfo.InvariantCulture));
  113. }
  114. return result.ToString();
  115. }
  116. }
  117. }