SimpleProtectedNegotiationToken.cs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. /* Copyright (C) 2017-2018 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
  2. *
  3. * You can redistribute this program and/or modify it under the terms of
  4. * the GNU Lesser Public License as published by the Free Software Foundation,
  5. * either version 3 of the License, or (at your option) any later version.
  6. */
  7. using System;
  8. using System.Collections.Generic;
  9. using Utilities;
  10. namespace SMBLibrary.Authentication.GSSAPI
  11. {
  12. public abstract class SimpleProtectedNegotiationToken
  13. {
  14. public const byte ApplicationTag = 0x60;
  15. public static readonly byte[] SPNEGOIdentifier = new byte[] { 0x2b, 0x06, 0x01, 0x05, 0x05, 0x02 };
  16. public abstract byte[] GetBytes();
  17. /// <param name="includeHeader">Prepend the generic GSSAPI header. Required for negTokenInit, optional for negTokenResp.</param>
  18. public byte[] GetBytes(bool includeHeader)
  19. {
  20. byte[] tokenBytes = this.GetBytes();
  21. if (includeHeader)
  22. {
  23. int objectIdentifierFieldSize = DerEncodingHelper.GetLengthFieldSize(SPNEGOIdentifier.Length);
  24. int tokenLength = 1 + objectIdentifierFieldSize + SPNEGOIdentifier.Length + tokenBytes.Length;
  25. int tokenLengthFieldSize = DerEncodingHelper.GetLengthFieldSize(tokenLength);
  26. int headerLength = 1 + tokenLengthFieldSize + 1 + objectIdentifierFieldSize + SPNEGOIdentifier.Length;
  27. byte[] buffer = new byte[headerLength + tokenBytes.Length];
  28. int offset = 0;
  29. ByteWriter.WriteByte(buffer, ref offset, ApplicationTag);
  30. DerEncodingHelper.WriteLength(buffer, ref offset, tokenLength);
  31. ByteWriter.WriteByte(buffer, ref offset, (byte)DerEncodingTag.ObjectIdentifier);
  32. DerEncodingHelper.WriteLength(buffer, ref offset, SPNEGOIdentifier.Length);
  33. ByteWriter.WriteBytes(buffer, ref offset, SPNEGOIdentifier);
  34. ByteWriter.WriteBytes(buffer, ref offset, tokenBytes);
  35. return buffer;
  36. }
  37. else
  38. {
  39. return tokenBytes;
  40. }
  41. }
  42. /// <summary>
  43. /// https://tools.ietf.org/html/rfc2743
  44. /// </summary>
  45. /// <exception cref="System.IO.InvalidDataException"></exception>
  46. public static SimpleProtectedNegotiationToken ReadToken(byte[] tokenBytes, int offset, bool serverInitiatedNegotiation)
  47. {
  48. byte tag = ByteReader.ReadByte(tokenBytes, ref offset);
  49. if (tag == ApplicationTag)
  50. {
  51. // https://msdn.microsoft.com/en-us/library/ms995330.aspx
  52. // when an InitToken is sent, it is prepended by an Application Constructed Object specifier (0x60),
  53. // and the OID for SPNEGO. This is the generic GSSAPI header.
  54. // [RFC 2743] The use of the Mechanism-Independent Token Format is required for initial context
  55. // establishment tokens, use in non-initial tokens is optional.
  56. int tokenLength = DerEncodingHelper.ReadLength(tokenBytes, ref offset);
  57. tag = ByteReader.ReadByte(tokenBytes, ref offset);
  58. if (tag == (byte)DerEncodingTag.ObjectIdentifier)
  59. {
  60. int objectIdentifierLength = DerEncodingHelper.ReadLength(tokenBytes, ref offset);
  61. byte[] objectIdentifier = ByteReader.ReadBytes(tokenBytes, ref offset, objectIdentifierLength);
  62. if (ByteUtils.AreByteArraysEqual(objectIdentifier, SPNEGOIdentifier))
  63. {
  64. tag = ByteReader.ReadByte(tokenBytes, ref offset);
  65. if (tag == SimpleProtectedNegotiationTokenInit.NegTokenInitTag)
  66. {
  67. if (serverInitiatedNegotiation)
  68. {
  69. // [MS-SPNG] Standard GSS has a strict notion of client (initiator) and server (acceptor).
  70. // If the client has not sent a negTokenInit ([RFC4178] section 4.2.1) message, no context establishment token is expected from the server.
  71. // The [NegTokenInit2] SPNEGO extension allows the server to generate a context establishment token message [..] and send it to the client.
  72. return new SimpleProtectedNegotiationTokenInit2(tokenBytes, offset);
  73. }
  74. else
  75. {
  76. return new SimpleProtectedNegotiationTokenInit(tokenBytes, offset);
  77. }
  78. }
  79. else if (tag == SimpleProtectedNegotiationTokenResponse.NegTokenRespTag)
  80. {
  81. return new SimpleProtectedNegotiationTokenResponse(tokenBytes, offset);
  82. }
  83. }
  84. }
  85. }
  86. else if (tag == SimpleProtectedNegotiationTokenResponse.NegTokenRespTag)
  87. {
  88. return new SimpleProtectedNegotiationTokenResponse(tokenBytes, offset);
  89. }
  90. return null;
  91. }
  92. }
  93. }