diff --git a/src/Accounts/Accounts/Account/ConnectAzureRmAccount.cs b/src/Accounts/Accounts/Account/ConnectAzureRmAccount.cs index 1a5a3536873d..4492ca84f1ff 100644 --- a/src/Accounts/Accounts/Account/ConnectAzureRmAccount.cs +++ b/src/Accounts/Accounts/Account/ConnectAzureRmAccount.cs @@ -45,6 +45,7 @@ using Microsoft.WindowsAzure.Commands.Utilities.Common; using Microsoft.Azure.PowerShell.Common.Share.Survey; using Microsoft.Azure.Commands.Profile.Utilities; +using System.Management.Automation.Runspaces; namespace Microsoft.Azure.Commands.Profile { @@ -249,6 +250,7 @@ protected override IAzureContext DefaultContext protected override void BeginProcessing() { base.BeginProcessing(); + ValidateActionRequiredMessageCanBePresented(); if (AzureEnvironment.PublicEnvironments.ContainsKey(EnvironmentName.AzureCloud)) { _environment = AzureEnvironment.PublicEnvironments[EnvironmentName.AzureCloud]; @@ -273,11 +275,19 @@ protected override void BeginProcessing() _writeWarningEvent -= WriteWarningSender; _writeWarningEvent += WriteWarningSender; + _writeInformationEvent -= WriteInformationSender; + _writeInformationEvent += WriteInformationSender; + // store the original write warning handler, register a thread safe one AzureSession.Instance.TryGetComponent(WriteWarningKey, out _originalWriteWarning); AzureSession.Instance.UnregisterComponent>(WriteWarningKey); AzureSession.Instance.RegisterComponent(WriteWarningKey, () => _writeWarningEvent); + // store the original write information handler, register a thread safe one + AzureSession.Instance.TryGetComponent(WriteInformationKey, out _originalWriteInformation); + AzureSession.Instance.UnregisterComponent>(WriteInformationKey); + AzureSession.Instance.RegisterComponent(WriteInformationKey, () => _writeInformationEvent); + // todo: ideally cancellation token should be passed to authentication factory as a parameter // however AuthenticationFactory.Authenticate does not support it // so I store it in AzureSession.Instance as a global variable @@ -289,11 +299,19 @@ protected override void BeginProcessing() private event EventHandler _writeWarningEvent; private event EventHandler _originalWriteWarning; + private event EventHandler _writeInformationEvent; + private event EventHandler _originalWriteInformation; + private void WriteWarningSender(object sender, StreamEventArgs args) { _tasks.Enqueue(new Task(() => this.WriteWarning(args.Message))); } + private void WriteInformationSender(object sender, StreamEventArgs args) + { + _tasks.Enqueue(new Task(() => this.WriteInformation(args.Message))); + } + protected override void StopProcessing() { if (AzureSession.Instance.TryGetComponent("LoginCancellationToken", out CancellationTokenSource cancellationTokenSource)) @@ -562,6 +580,20 @@ public override void ExecuteCmdlet() } } + private void ValidateActionRequiredMessageCanBePresented() + { + if (UseDeviceAuthentication.IsPresent && IsWriteInformationIgnored()) + { + throw new ActionPreferenceStopException(Resources.DoNotIgnoreInformationIfUserDeviceAuth); + } + } + + private bool IsWriteInformationIgnored() + { + return !MyInvocation.BoundParameters.ContainsKey("InformationAction") && ActionPreference.Ignore.ToString().Equals(SessionState?.PSVariable?.GetValue("InformationPreference", ActionPreference.SilentlyContinue)?.ToString() ?? "") || + MyInvocation.BoundParameters.TryGetValue("InformationAction", out var value) && ActionPreference.Ignore.ToString().Equals(value?.ToString() ?? "", StringComparison.InvariantCultureIgnoreCase); + } + private string PreProcessAuthScope() { string mappedScope = AuthScope; @@ -774,6 +806,9 @@ protected override void EndProcessing() // unregister the thread-safe write warning, because it won't work out of this cmdlet AzureSession.Instance.UnregisterComponent>(WriteWarningKey); AzureSession.Instance.RegisterComponent(WriteWarningKey, () => _originalWriteWarning); + // unregister the thread-safe write information, because it won't work out of this cmdlet + AzureSession.Instance.UnregisterComponent>(WriteInformationKey); + AzureSession.Instance.RegisterComponent(WriteInformationKey, () => _originalWriteInformation); } } } diff --git a/src/Accounts/Accounts/ChangeLog.md b/src/Accounts/Accounts/ChangeLog.md index aef2a77c40ee..84e90958eb29 100644 --- a/src/Accounts/Accounts/ChangeLog.md +++ b/src/Accounts/Accounts/ChangeLog.md @@ -19,12 +19,15 @@ --> ## Upcoming Release +* Redirected device code login messages from warning stream to information stream if use device authentication in `Connect-AzAccount`. +* Adjusted output format to be more user-friendly for `Get-AzContext/Tenant/Subscription` and `Invoke-AzRestMethod`, including + - ordering and grouping output items to make items easy to find. + - re-prioritizing positions for output properties to highlight valuable properties. * Upgraded the reference of Azure PowerShell Common to 1.3.90-preview. * Upgraded Azure.Identity to 1.10.3 [#23018]. - Renamed token cache from `msal.cache` to `msal.cache.cae` or `masl.cache.nocae`. * Enabled Continue Access Evaluation (CAE) for all Service Principals login methods. * Supported signing in with Microsoft Account (MSA) via Web Account Manager (WAM). Enable it by `Set-AzConfig -EnableLoginByWam $true`. -* Adjusted output format to be more user-friendly for `Get-AzContext/Tenant/Subscription` and `Invoke-AzRestMethod`. * Fixed the multiple `x-ms-unique-id` values issue. ## Version 2.15.0 diff --git a/src/Accounts/Accounts/Properties/Resources.Designer.cs b/src/Accounts/Accounts/Properties/Resources.Designer.cs index 2298c200f30a..1e31ef0f240a 100644 --- a/src/Accounts/Accounts/Properties/Resources.Designer.cs +++ b/src/Accounts/Accounts/Properties/Resources.Designer.cs @@ -537,6 +537,15 @@ internal static string DisableDataCollection { } } + /// + /// Looks up a localized string similar to Please do not set InformationAction or $InformationPreference to Ignore if you want to use device code authentication.. + /// + internal static string DoNotIgnoreInformationIfUserDeviceAuth { + get { + return ResourceManager.GetString("DoNotIgnoreInformationIfUserDeviceAuth", resourceCulture); + } + } + /// /// Looks up a localized string similar to Allow Azure PowerShell cmdlets to send data to Microsoft to improve the customer experience. /// diff --git a/src/Accounts/Accounts/Properties/Resources.resx b/src/Accounts/Accounts/Properties/Resources.resx index a8b3f3f8e0cd..f3d5dceb694b 100644 --- a/src/Accounts/Accounts/Properties/Resources.resx +++ b/src/Accounts/Accounts/Properties/Resources.resx @@ -595,4 +595,7 @@ Personally identifiable information and confidential data may be written to the file located at '{0}'. Please ensure that appropriate access controls are assigned to the saved file. + + Please do not set InformationAction or $InformationPreference to Ignore if you want to use device code authentication. + \ No newline at end of file diff --git a/src/Accounts/Authenticators/DeviceCodeAuthenticator.cs b/src/Accounts/Authenticators/DeviceCodeAuthenticator.cs index e615e4758df2..4bba654f0262 100644 --- a/src/Accounts/Authenticators/DeviceCodeAuthenticator.cs +++ b/src/Accounts/Authenticators/DeviceCodeAuthenticator.cs @@ -12,10 +12,6 @@ // limitations under the License. // ---------------------------------------------------------------------------------- -using System; -using System.Threading; -using System.Threading.Tasks; - using Azure.Core; using Azure.Identity; @@ -24,6 +20,12 @@ using Microsoft.Azure.Commands.Common.Authentication; using Microsoft.Azure.Commands.Common.Authentication.Abstractions; using Microsoft.Azure.Commands.ResourceManager.Common; +using Microsoft.WindowsAzure.Commands.Common; + +using System; +using System.Text; +using System.Threading; +using System.Threading.Tasks; namespace Microsoft.Azure.PowerShell.Authenticators { @@ -64,7 +66,7 @@ public override Task Authenticate(AuthenticationParameters paramet private Task DeviceCodeFunc(DeviceCodeInfo info, CancellationToken cancellation) { - WriteWarning(info.Message); + WriteInfomartion(info.Message, info.UserCode); return Task.CompletedTask; } @@ -73,12 +75,29 @@ public override bool CanAuthenticate(AuthenticationParameters parameters) return (parameters as DeviceCodeParameters) != null; } - private void WriteWarning(string message) + + private void WriteInfomartion(string message, string userCode) { - EventHandler writeWarningEvent; - if (AzureSession.Instance.TryGetComponent(AzureRMCmdlet.WriteWarningKey, out writeWarningEvent)) + + var loginInfo = new StringBuilder(); + string LoginToAzurePhrase = $"{PSStyle.Bold}{PSStyle.BackgroundColor.Blue}[Login to Azure]{PSStyle.Reset} "; + loginInfo.Append(LoginToAzurePhrase); + + if (!string.IsNullOrEmpty(userCode)) + { + var formattedUserCode = $"{PSStyle.Underline}{userCode}{PSStyle.Reset}"; + var formattedMessage = message.Replace(userCode, formattedUserCode); + loginInfo.Append(formattedMessage); + } + else + { + loginInfo.Append(message); + } + + EventHandler writeInforamtionEvent; + if (AzureSession.Instance.TryGetComponent(AzureRMCmdlet.WriteInformationKey, out writeInforamtionEvent)) { - writeWarningEvent(this, new StreamEventArgs() { Message = message }); + writeInforamtionEvent(this, new StreamEventArgs() { Message = loginInfo.ToString() }); } } }