using System; using System.Linq; using System.Text; using System.Threading; using Renci.SshNet.Common; using Renci.SshNet.Messages.Authentication; using Renci.SshNet.Messages; namespace Renci.SshNet { /// /// Provides functionality to perform password authentication. /// public partial class PasswordAuthenticationMethod : AuthenticationMethod, IDisposable { private AuthenticationResult _authenticationResult = AuthenticationResult.Failure; private Session _session; private EventWaitHandle _authenticationCompleted = new AutoResetEvent(false); private Exception _exception; private readonly RequestMessage _requestMessage; private readonly byte[] _password; /// /// Gets authentication method name /// public override string Name { get { return this._requestMessage.MethodName; } } /// /// Occurs when user's password has expired and needs to be changed. /// public event EventHandler PasswordExpired; /// /// Initializes a new instance of the class. /// /// The username. /// The password. /// is whitespace or null. /// is null. public PasswordAuthenticationMethod(string username, string password) : this(username, Encoding.UTF8.GetBytes(password)) { } /// /// Initializes a new instance of the class. /// /// The username. /// The password. /// is whitespace or null. /// is null. public PasswordAuthenticationMethod(string username, byte[] password) : base(username) { if (password == null) throw new ArgumentNullException("password"); this._password = password; this._requestMessage = new RequestMessagePassword(ServiceName.Connection, this.Username, this._password); } /// /// Authenticates the specified session. /// /// The session to authenticate. /// /// Result of authentication process. /// /// is null. public override AuthenticationResult Authenticate(Session session) { if (session == null) throw new ArgumentNullException("session"); this._session = session; session.UserAuthenticationSuccessReceived += Session_UserAuthenticationSuccessReceived; session.UserAuthenticationFailureReceived += Session_UserAuthenticationFailureReceived; session.MessageReceived += Session_MessageReceived; session.RegisterMessage("SSH_MSG_USERAUTH_PASSWD_CHANGEREQ"); session.SendMessage(this._requestMessage); session.WaitOnHandle(this._authenticationCompleted); session.UserAuthenticationSuccessReceived -= Session_UserAuthenticationSuccessReceived; session.UserAuthenticationFailureReceived -= Session_UserAuthenticationFailureReceived; session.MessageReceived -= Session_MessageReceived; if (this._exception != null) { throw this._exception; } return this._authenticationResult; } private void Session_UserAuthenticationSuccessReceived(object sender, MessageEventArgs e) { this._authenticationResult = AuthenticationResult.Success; this._authenticationCompleted.Set(); } private void Session_UserAuthenticationFailureReceived(object sender, MessageEventArgs e) { if (e.Message.PartialSuccess) this._authenticationResult = AuthenticationResult.PartialSuccess; else this._authenticationResult = AuthenticationResult.Failure; // Copy allowed authentication methods this.AllowedAuthentications = e.Message.AllowedAuthentications.ToList(); this._authenticationCompleted.Set(); } private void Session_MessageReceived(object sender, MessageEventArgs e) { if (e.Message is PasswordChangeRequiredMessage) { this._session.UnRegisterMessage("SSH_MSG_USERAUTH_PASSWD_CHANGEREQ"); this.ExecuteThread(() => { try { var eventArgs = new AuthenticationPasswordChangeEventArgs(this.Username); // Raise an event to allow user to supply a new password if (this.PasswordExpired != null) { this.PasswordExpired(this, eventArgs); } // Send new authentication request with new password this._session.SendMessage(new RequestMessagePassword(ServiceName.Connection, this.Username, this._password, eventArgs.NewPassword)); } catch (Exception exp) { this._exception = exp; this._authenticationCompleted.Set(); } }); } } partial void ExecuteThread(Action action); #region IDisposable Members private bool _isDisposed; /// /// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources. /// public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } /// /// Releases unmanaged and - optionally - managed resources /// /// true to release both managed and unmanaged resources; false to release only unmanaged resources. protected virtual void Dispose(bool disposing) { // Check to see if Dispose has already been called. if (!this._isDisposed) { // If disposing equals true, dispose all managed // and unmanaged resources. if (disposing) { // Dispose managed resources. if (this._authenticationCompleted != null) { this._authenticationCompleted.Dispose(); this._authenticationCompleted = null; } } // Note disposing has been done. _isDisposed = true; } } /// /// Releases unmanaged resources and performs other cleanup operations before the /// is reclaimed by garbage collection. /// ~PasswordAuthenticationMethod() { // Do not re-create Dispose clean-up code here. // Calling Dispose(false) is optimal in terms of // readability and maintainability. Dispose(false); } #endregion } }