123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213 |
- /* Copyright (C) 2017 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
- *
- * You can redistribute this program and/or modify it under the terms of
- * the GNU Lesser Public License as published by the Free Software Foundation,
- * either version 3 of the License, or (at your option) any later version.
- */
- using System;
- using System.Collections.Generic;
- using SMBLibrary.Authentication.NTLM;
- using Utilities;
- namespace SMBLibrary.Authentication.GSSAPI
- {
- public class GSSProvider
- {
- public static readonly byte[] NTLMSSPIdentifier = new byte[] { 0x2b, 0x06, 0x01, 0x04, 0x01, 0x82, 0x37, 0x02, 0x02, 0x0a };
- private List<IGSSMechanism> m_mechanisms;
- private Dictionary<object, IGSSMechanism> m_contextToMechanism = new Dictionary<object, IGSSMechanism>();
- public GSSProvider(IGSSMechanism mechanism)
- {
- m_mechanisms = new List<IGSSMechanism>();
- m_mechanisms.Add(mechanism);
- }
- public GSSProvider(List<IGSSMechanism> mechanisms)
- {
- m_mechanisms = mechanisms;
- }
- public byte[] GetSPNEGOTokenInitBytes()
- {
- SimpleProtectedNegotiationTokenInit token = new SimpleProtectedNegotiationTokenInit();
- token.MechanismTypeList = new List<byte[]>();
- foreach (IGSSMechanism mechanism in m_mechanisms)
- {
- token.MechanismTypeList.Add(mechanism.Identifier);
- }
- return SimpleProtectedNegotiationToken.GetTokenBytes(token);
- }
- public NTStatus AcceptSecurityContext(ref object context, byte[] inputToken, out byte[] outputToken)
- {
- outputToken = null;
- SimpleProtectedNegotiationToken spnegoToken = SimpleProtectedNegotiationToken.ReadToken(inputToken, 0);
- if (spnegoToken != null)
- {
- if (spnegoToken is SimpleProtectedNegotiationTokenInit)
- {
- SimpleProtectedNegotiationTokenInit tokenInit = (SimpleProtectedNegotiationTokenInit)spnegoToken;
- IGSSMechanism mechanism = FindMechanism(tokenInit.MechanismTypeList);
- if (mechanism != null)
- {
- byte[] mechanismOutput;
- NTStatus status = mechanism.AcceptSecurityContext(ref context, tokenInit.MechanismToken, out mechanismOutput);
- outputToken = GetSPNEGOTokenResponseBytes(mechanismOutput, status, mechanism.Identifier);
- m_contextToMechanism[context] = mechanism;
- return status;
- }
- return NTStatus.SEC_E_SECPKG_NOT_FOUND;
- }
- else // SimpleProtectedNegotiationTokenResponse
- {
- IGSSMechanism mechanism;
- if (!m_contextToMechanism.TryGetValue(context, out mechanism))
- {
- // We assume that the problem is not with our implementation and that the client has sent
- // SimpleProtectedNegotiationTokenResponse without first sending SimpleProtectedNegotiationTokenInit.
- return NTStatus.SEC_E_INVALID_TOKEN;
- }
- SimpleProtectedNegotiationTokenResponse tokenResponse = (SimpleProtectedNegotiationTokenResponse)spnegoToken;
- byte[] mechanismOutput;
- NTStatus status = mechanism.AcceptSecurityContext(ref context, tokenResponse.ResponseToken, out mechanismOutput);
- outputToken = GetSPNEGOTokenResponseBytes(mechanismOutput, status, null);
- return status;
- }
- }
- else
- {
- // [MS-SMB] The Windows GSS implementation supports raw Kerberos / NTLM messages in the SecurityBlob.
- // [MS-SMB2] Windows [..] will also accept raw Kerberos messages and implicit NTLM messages as part of GSS authentication.
- if (AuthenticationMessageUtils.IsSignatureValid(inputToken))
- {
- MessageTypeName messageType = AuthenticationMessageUtils.GetMessageType(inputToken);
- IGSSMechanism ntlmAuthenticationProvider = FindMechanism(NTLMSSPIdentifier);
- if (ntlmAuthenticationProvider != null)
- {
- NTStatus status = ntlmAuthenticationProvider.AcceptSecurityContext(ref context, inputToken, out outputToken);
- if (messageType == MessageTypeName.Negotiate)
- {
- m_contextToMechanism[context] = ntlmAuthenticationProvider;
- }
- return status;
- }
- else
- {
- return NTStatus.SEC_E_SECPKG_NOT_FOUND;
- }
- }
- }
- return NTStatus.SEC_E_INVALID_TOKEN;
- }
- public object GetContextAttribute(object context, GSSAttributeName attributeName)
- {
- IGSSMechanism mechanism;
- if (!m_contextToMechanism.TryGetValue(context, out mechanism))
- {
- return null;
- }
- return mechanism.GetContextAttribute(context, attributeName);
- }
- public void DeleteSecurityContext(ref object context)
- {
- if (context != null)
- {
- IGSSMechanism mechanism;
- if (m_contextToMechanism.TryGetValue(context, out mechanism))
- {
- mechanism.DeleteSecurityContext(ref context);
- m_contextToMechanism.Remove(context);
- }
- }
- }
- /// <summary>
- /// Helper method for legacy implementation.
- /// </summary>
- public NTStatus GetNTLMChallengeMessage(out object context, NegotiateMessage negotiateMessage, out ChallengeMessage challengeMessage)
- {
- context = null;
- challengeMessage = null;
- IGSSMechanism ntlmAuthenticationProvider = FindMechanism(NTLMSSPIdentifier);
- if (ntlmAuthenticationProvider != null)
- {
- byte[] outputToken;
- NTStatus result = ntlmAuthenticationProvider.AcceptSecurityContext(ref context, negotiateMessage.GetBytes(), out outputToken);
- challengeMessage = new ChallengeMessage(outputToken);
- m_contextToMechanism.Add(context, ntlmAuthenticationProvider);
- return result;
- }
- else
- {
- return NTStatus.SEC_E_SECPKG_NOT_FOUND;
- }
- }
- /// <summary>
- /// Helper method for legacy implementation.
- /// </summary>
- public NTStatus NTLMAuthenticate(object context, AuthenticateMessage authenticateMessage)
- {
- IGSSMechanism ntlmAuthenticationProvider = FindMechanism(NTLMSSPIdentifier);
- if (ntlmAuthenticationProvider != null)
- {
- byte[] outputToken;
- NTStatus result = ntlmAuthenticationProvider.AcceptSecurityContext(ref context, authenticateMessage.GetBytes(), out outputToken);
- return result;
- }
- else
- {
- return NTStatus.SEC_E_SECPKG_NOT_FOUND;
- }
- }
- public IGSSMechanism FindMechanism(List<byte[]> mechanismIdentifiers)
- {
- foreach (byte[] identifier in mechanismIdentifiers)
- {
- IGSSMechanism mechanism = FindMechanism(identifier);
- if (mechanism != null)
- {
- return mechanism;
- }
- }
- return null;
- }
- public IGSSMechanism FindMechanism(byte[] mechanismIdentifier)
- {
- foreach (IGSSMechanism mechanism in m_mechanisms)
- {
- if (ByteUtils.AreByteArraysEqual(mechanism.Identifier, mechanismIdentifier))
- {
- return mechanism;
- }
- }
- return null;
- }
- private static byte[] GetSPNEGOTokenResponseBytes(byte[] mechanismOutput, NTStatus status, byte[] mechanismIdentifier)
- {
- SimpleProtectedNegotiationTokenResponse tokenResponse = new SimpleProtectedNegotiationTokenResponse();
- if (status == NTStatus.STATUS_SUCCESS)
- {
- tokenResponse.NegState = NegState.AcceptCompleted;
- }
- else if (status == NTStatus.SEC_I_CONTINUE_NEEDED)
- {
- tokenResponse.NegState = NegState.AcceptIncomplete;
- }
- else
- {
- tokenResponse.NegState = NegState.Reject;
- }
- tokenResponse.SupportedMechanism = mechanismIdentifier;
- tokenResponse.ResponseToken = mechanismOutput;
- return tokenResponse.GetBytes();
- }
- }
- }
|