Skip to content

Commit 4248c1d

Browse files
gautamdshethGautam Sheth
andauthored
Add PSVersionUtility to retrieve PowerShell version and update cmdlets for admin checks (#4788)
* Bump MSAL to 4.69 * Add PSVersionUtility to retrieve PowerShell version and update cmdlets for admin checks * Refactor admin check logic to use PSUtility for local admin verification and remove deprecated PSVersionUtility class * Update admin check logic to throw exceptions instead of warnings and refine local admin verification in PSUtility * Update error message for local admin requirement to clarify .NET 9 bug * Refactor CertificateHelper and PSUtility for improved logging and admin check logic * Update error messages for local admin requirement to clarify elevated permissions needed * Update error messages to specify elevated permissions required for certificate generation --------- Co-authored-by: Gautam Sheth <gautam.sheth@staffbase.com>
1 parent 4f7c933 commit 4248c1d

File tree

6 files changed

+93
-53
lines changed

6 files changed

+93
-53
lines changed

src/Commands/AzureAD/RegisterAzureADApp.cs

Lines changed: 9 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,12 @@
11
using PnP.Framework;
2+
using PnP.PowerShell.Commands.Base;
3+
using PnP.PowerShell.Commands.Enums;
24
using PnP.PowerShell.Commands.Model;
35
using PnP.PowerShell.Commands.Utilities;
46
using PnP.PowerShell.Commands.Utilities.REST;
57
using System;
68
using System.Collections.Generic;
9+
using System.Dynamic;
710
using System.IO;
811
using System.Linq;
912
using System.Management.Automation;
@@ -13,12 +16,9 @@
1316
using System.Security.Cryptography.X509Certificates;
1417
using System.Threading;
1518
using System.Threading.Tasks;
19+
using TextCopy;
1620
using OperatingSystem = PnP.PowerShell.Commands.Utilities.OperatingSystem;
1721
using Resources = PnP.PowerShell.Commands.Properties.Resources;
18-
using PnP.PowerShell.Commands.Base;
19-
using System.Dynamic;
20-
using PnP.PowerShell.Commands.Enums;
21-
using TextCopy;
2222

2323
namespace PnP.PowerShell.Commands.AzureAD
2424
{
@@ -94,6 +94,11 @@ public class RegisterAzureADApp : BasePSCmdlet, IDynamicParameters
9494

9595
protected override void ProcessRecord()
9696
{
97+
if (!PSUtility.IsUserLocalAdmin())
98+
{
99+
throw new PSArgumentException("Running this cmdlet requires elevated permissions (Run as Admin) to generate a certificate.");
100+
}
101+
97102
if (ParameterSpecified(nameof(Store)) && !OperatingSystem.IsWindows())
98103
{
99104
throw new PSArgumentException("The Store parameter is only supported on Microsoft Windows");

src/Commands/AzureAD/RegisterEntraIDAppForInteractiveLogin.cs

Lines changed: 11 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,20 @@
11
using PnP.Framework;
2+
using PnP.PowerShell.Commands.Base;
3+
using PnP.PowerShell.Commands.Enums;
24
using PnP.PowerShell.Commands.Model;
35
using PnP.PowerShell.Commands.Utilities;
46
using PnP.PowerShell.Commands.Utilities.REST;
57
using System;
68
using System.Collections.Generic;
9+
using System.Dynamic;
710
using System.IO;
811
using System.Linq;
912
using System.Management.Automation;
1013
using System.Net.Http;
1114
using System.Threading;
1215
using System.Threading.Tasks;
13-
using OperatingSystem = PnP.PowerShell.Commands.Utilities.OperatingSystem;
14-
using PnP.PowerShell.Commands.Base;
15-
using System.Dynamic;
16-
using PnP.PowerShell.Commands.Enums;
1716
using TextCopy;
17+
using OperatingSystem = PnP.PowerShell.Commands.Utilities.OperatingSystem;
1818

1919
namespace PnP.PowerShell.Commands.AzureAD
2020
{
@@ -49,6 +49,11 @@ public class RegisterEntraIDAppForInteractiveLogin : BasePSCmdlet, IDynamicParam
4949

5050
protected override void ProcessRecord()
5151
{
52+
if (!PSUtility.IsUserLocalAdmin())
53+
{
54+
throw new PSArgumentException("Running this cmdlet requires elevated permissions (Run as Admin) to generate a certificate.");
55+
}
56+
5257
var redirectUri = "http://localhost";
5358
// if (ParameterSpecified(nameof(DeviceLogin)) || OperatingSystem.IsMacOS())
5459
if (ParameterSpecified(nameof(DeviceLogin)) || OperatingSystem.IsMacOS())
@@ -460,7 +465,7 @@ private AzureADApp CreateApp(string loginEndPoint, HttpClient httpClient, string
460465
private void StartConsentFlow(string loginEndPoint, AzureADApp azureApp, string redirectUri, string token, HttpClient httpClient, PSObject record, CmdletMessageWriter messageWriter, List<PermissionScope> scopes)
461466
{
462467
var htmlMessageConsentSuccess = $"<html lang=en><meta charset=utf-8><title>PnP PowerShell - Consent</title><meta content=\"width=device-width,initial-scale=1\"name=viewport><style>html{{height:100%}}.message-container{{flex-grow:1;display:flex;align-items:center;justify-content:center;margin:0 30px}}body{{box-sizing:border-box;min-height:100%;display:flex;flex-direction:column;color:#fff;font-family:\"Segoe UI\",\"Helvetica Neue\",Helvetica,Arial,sans-serif;background-color:#2c2c32;margin:0;padding:15px 30px}}.message{{font-weight:300;font-size:1.4rem}}.branding{{background-image:url();background-repeat:no-repeat;padding-left:26px;font-size:20px;letter-spacing:-.04rem;font-weight:400;height:26px;color:#fff;background-position:left center;text-decoration:none}}</style><a class=branding href=https://pnp.github.io/powershell>PnP PowerShell</a><div class=message-container><div class=message>You successfully provided consent now and can close this page.</div></div>";
463-
var htmlMessageConsentFailed= $"<html lang=en><meta charset=utf-8><title>PnP PowerShell - Consent</title><meta content=\"width=device-width,initial-scale=1\"name=viewport><style>html{{height:100%}}.error-text{{color:red;font-size:1rem}}.message-container{{flex-grow:1;display:flex;align-items:center;justify-content:center;margin:0 30px}}body{{box-sizing:border-box;min-height:100%;display:flex;flex-direction:column;color:#fff;font-family:\"Segoe UI\",\"Helvetica Neue\",Helvetica,Arial,sans-serif;background-color:#2c2c32;margin:0;padding:15px 30px}}.message{{font-weight:300;font-size:1.4rem}}.branding{{background-image:url();background-repeat:no-repeat;height:26px;padding-left:26px;font-size:20px;letter-spacing:-.04rem;font-weight:400;color:#fff;background-position:left center;text-decoration:none}}</style><a class=branding href=https://pnp.github.io/powershell>PnP PowerShell</a><div class=message-container><div class=message>You failed to provide consent. Please try again. You can close this page.</div></div>";
468+
var htmlMessageConsentFailed = $"<html lang=en><meta charset=utf-8><title>PnP PowerShell - Consent</title><meta content=\"width=device-width,initial-scale=1\"name=viewport><style>html{{height:100%}}.error-text{{color:red;font-size:1rem}}.message-container{{flex-grow:1;display:flex;align-items:center;justify-content:center;margin:0 30px}}body{{box-sizing:border-box;min-height:100%;display:flex;flex-direction:column;color:#fff;font-family:\"Segoe UI\",\"Helvetica Neue\",Helvetica,Arial,sans-serif;background-color:#2c2c32;margin:0;padding:15px 30px}}.message{{font-weight:300;font-size:1.4rem}}.branding{{background-image:url();background-repeat:no-repeat;height:26px;padding-left:26px;font-size:20px;letter-spacing:-.04rem;font-weight:400;color:#fff;background-position:left center;text-decoration:none}}</style><a class=branding href=https://pnp.github.io/powershell>PnP PowerShell</a><div class=message-container><div class=message>You failed to provide consent. Please try again. You can close this page.</div></div>";
464469

465470
var graphEndpoint = $"https://{AuthenticationManager.GetGraphEndPoint(AzureEnvironment)}";
466471
if (AzureEnvironment == AzureEnvironment.Custom)
@@ -598,7 +603,7 @@ private void SetLogo(AzureADApp azureApp, string token)
598603
{
599604
var byteArrayContent = new ByteArrayContent(bytes);
600605
byteArrayContent.Headers.ContentType = new System.Net.Http.Headers.MediaTypeHeaderValue(mediaType);
601-
var requestHelper = new ApiRequestHelper(GetType(),PnPConnection.Current);
606+
var requestHelper = new ApiRequestHelper(GetType(), PnPConnection.Current);
602607
requestHelper.Put2(endpoint, byteArrayContent, token);
603608

604609
WriteVerbose("Successfully set the logo for the Entra ID app");

src/Commands/Base/NewAzureCertificate.cs

Lines changed: 6 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -54,14 +54,19 @@ protected override void ProcessRecord()
5454
throw new PSArgumentException("The Store parameter is only supported on Microsoft Windows");
5555
}
5656

57+
if (!PSUtility.IsUserLocalAdmin())
58+
{
59+
throw new PSArgumentException("Running this cmdlet requires elevated permissions (Run as Admin) to generate a certificate.");
60+
}
61+
5762
if (ValidYears < 1 || ValidYears > 30)
5863
{
5964
ValidYears = 10;
6065
}
6166
DateTime validFrom = DateTime.Today;
6267
DateTime validTo = validFrom.AddYears(ValidYears);
6368

64-
if(MyInvocation.BoundParameters.ContainsKey(nameof(SanNames)) && SanNames == null)
69+
if (MyInvocation.BoundParameters.ContainsKey(nameof(SanNames)) && SanNames == null)
6570
{
6671
SanNames = Array.Empty<string>();
6772
}

src/Commands/Base/PnPConnection.cs

Lines changed: 1 addition & 30 deletions
Original file line numberDiff line numberDiff line change
@@ -847,40 +847,11 @@ internal void InitializeTelemetry(ClientContext context, InitializationType init
847847
var coreAssembly = Assembly.GetExecutingAssembly();
848848
var operatingSystem = Utilities.OperatingSystem.GetOSString();
849849

850-
ApplicationInsights.Initialize(serverLibraryVersion, serverVersion, initializationType.ToString(), ((AssemblyFileVersionAttribute)coreAssembly.GetCustomAttribute(typeof(AssemblyFileVersionAttribute))).Version.ToString(), operatingSystem, PSVersion);
850+
ApplicationInsights.Initialize(serverLibraryVersion, serverVersion, initializationType.ToString(), ((AssemblyFileVersionAttribute)coreAssembly.GetCustomAttribute(typeof(AssemblyFileVersionAttribute))).Version.ToString(), operatingSystem, PSUtility.PSVersion);
851851
ApplicationInsights.TrackEvent("Connect-PnPOnline");
852852
}
853853
}
854854

855-
private static string PSVersion => (PSVersionLazy.Value);
856-
857-
private static readonly Lazy<string> PSVersionLazy = new Lazy<string>(
858-
() =>
859-
860-
{
861-
var caller = AppDomain.CurrentDomain.GetAssemblies().SingleOrDefault(a => a.GetName().Name == "System.Management.Automation");
862-
//var caller = Assembly.GetCallingAssembly();
863-
var psVersionType = caller.GetType("System.Management.Automation.PSVersionInfo");
864-
if (null != psVersionType)
865-
{
866-
PropertyInfo propInfo = psVersionType.GetProperty("PSVersion");
867-
if (null == propInfo)
868-
{
869-
propInfo = psVersionType.GetProperty("PSVersion", BindingFlags.NonPublic | BindingFlags.Static);
870-
}
871-
var getter = propInfo.GetGetMethod(true);
872-
var version = getter.Invoke(null, new object[] { });
873-
874-
if (null != version)
875-
{
876-
var versionType = version.GetType();
877-
var versionProperty = versionType.GetProperty("Major");
878-
return ((int)versionProperty.GetValue(version)).ToString();
879-
}
880-
}
881-
return "";
882-
});
883-
884855
private static string PnPPSVersionTag => (PnPPSVersionTagLazy.Value);
885856

886857
private static readonly Lazy<string> PnPPSVersionTagLazy = new Lazy<string>(

0 commit comments

Comments
 (0)