From 79c682c4ecd9e24c273b074c89c8eb4192a0cbfa Mon Sep 17 00:00:00 2001 From: Teo Voinea Date: Thu, 4 Aug 2022 17:57:53 +0000 Subject: [PATCH 1/3] Add extensions functionality --- src/ApiService/ApiService/Program.cs | 1 + .../ApiService/onefuzzlib/Extension.cs | 216 +++++++++++++----- .../ApiService/onefuzzlib/ImageOperations.cs | 115 ++++++++++ .../ApiService/onefuzzlib/OnefuzzContext.cs | 2 + .../ApiService/onefuzzlib/Reports.cs | 11 + .../onefuzzlib/VmExtensionWrapper.cs | 62 +++++ .../ApiService/onefuzzlib/orm/Orm.cs | 3 + .../IntegrationTests/Fakes/TestContext.cs | 2 + 8 files changed, 358 insertions(+), 54 deletions(-) create mode 100644 src/ApiService/ApiService/onefuzzlib/ImageOperations.cs create mode 100644 src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs diff --git a/src/ApiService/ApiService/Program.cs b/src/ApiService/ApiService/Program.cs index 4d2c527fd2..c021210c48 100644 --- a/src/ApiService/ApiService/Program.cs +++ b/src/ApiService/ApiService/Program.cs @@ -109,6 +109,7 @@ public async static Async.Task Main() { .AddScoped() .AddScoped() .AddScoped() + .AddScoped() .AddScoped() .AddScoped() .AddScoped() diff --git a/src/ApiService/ApiService/onefuzzlib/Extension.cs b/src/ApiService/ApiService/onefuzzlib/Extension.cs index cb24b53e35..431f608e1a 100644 --- a/src/ApiService/ApiService/onefuzzlib/Extension.cs +++ b/src/ApiService/ApiService/onefuzzlib/Extension.cs @@ -1,4 +1,6 @@ using System.Text.Json; +using System.Threading.Tasks; +using Azure.Core; using Azure.ResourceManager.Compute; using Azure.Storage.Sas; using Microsoft.OneFuzz.Service.OneFuzzLib.Orm; @@ -7,8 +9,9 @@ namespace Microsoft.OneFuzz.Service; public interface IExtensions { public Async.Task> FuzzExtensions(Pool pool, Scaleset scaleset); -} + public Async.Task> ReproExtensions(AzureLocation region, Os reproOs, Guid reproId, ReproConfig reproConfig, Container? setupContainer); +} public class Extensions : IExtensions { IServiceConfig _serviceConfig; @@ -18,13 +21,20 @@ public class Extensions : IExtensions { IConfigOperations _instanceConfigOps; ILogAnalytics _logAnalytics; - public Extensions(IServiceConfig config, ICreds creds, IQueue queue, IContainers containers, IConfigOperations instanceConfigOps, ILogAnalytics logAnalytics) { + IOnefuzzContext _context; + + private static readonly JsonSerializerOptions _extensionSerializerOptions = new JsonSerializerOptions { + PropertyNamingPolicy = JsonNamingPolicy.CamelCase + }; + + public Extensions(IServiceConfig config, ICreds creds, IQueue queue, IContainers containers, IConfigOperations instanceConfigOps, ILogAnalytics logAnalytics, IOnefuzzContext context) { _serviceConfig = config; _creds = creds; _queue = queue; _containers = containers; _instanceConfigOps = instanceConfigOps; _logAnalytics = logAnalytics; + _context = context; } public async Async.Task ConfigUrl(Container container, string fileName, bool withSas) { @@ -34,8 +44,8 @@ public Extensions(IServiceConfig config, ICreds creds, IQueue queue, IContainers return await _containers.GetFileUrl(container, fileName, StorageType.Config); } - public async Async.Task> GenericExtensions(string region, Os vmOs) { - var extensions = new List(); + public async Async.Task> GenericExtensions(AzureLocation region, Os vmOs) { + var extensions = new List(); var instanceConfig = await _instanceConfigOps.Fetch(); extensions.Add(await MonitorExtension(region, vmOs)); @@ -71,19 +81,19 @@ public async Async.Task> GenericExten return extensions; } - public static VirtualMachineScaleSetExtensionData KeyVaultExtension(string region, KeyvaultExtensionConfig keyVault, Os vmOs) { + public static VMExtenionWrapper KeyVaultExtension(AzureLocation region, KeyvaultExtensionConfig keyVault, Os vmOs) { var keyVaultName = keyVault.KeyVaultName; var certName = keyVault.CertName; var uri = keyVaultName + certName; if (vmOs == Os.Windows) { - return new VirtualMachineScaleSetExtensionData { - Name = "KVVMExtensionForWindows", + return new VMExtenionWrapper { + Location = region, Publisher = "Microsoft.Azure.KeyVault", TypePropertiesType = "KeyVaultForWindows", TypeHandlerVersion = "1.0", AutoUpgradeMinorVersion = true, - Settings = new BinaryData(new { + Settings = new BinaryData(JsonSerializer.Serialize(new { SecretsManagementSettings = new { PollingIntervalInS = "3600", CertificateStoreName = "MY", @@ -92,46 +102,46 @@ public static VirtualMachineScaleSetExtensionData KeyVaultExtension(string regio RequireInitialSync = true, ObservedCertificates = new string[] { uri }, } - }) + }, _extensionSerializerOptions)) }; } else if (vmOs == Os.Linux) { var certPath = keyVault.CertPath; var extensionStore = keyVault.ExtensionStore; var certLocation = certPath + extensionStore; - return new VirtualMachineScaleSetExtensionData { - Name = "KVVMExtensionForLinux", + return new VMExtenionWrapper { + Location = region, Publisher = "Microsoft.Azure.KeyVault", TypePropertiesType = "KeyVaultForLinux", TypeHandlerVersion = "2.0", AutoUpgradeMinorVersion = true, - Settings = new BinaryData(new { + Settings = new BinaryData(JsonSerializer.Serialize(new { SecretsManagementSettings = new { PollingIntervalInS = "3600", CertificateStoreLocation = certLocation, RequireInitialSync = true, ObservedCertificates = new string[] { uri }, } - }) + }, _extensionSerializerOptions)) }; } else { throw new NotImplementedException($"unsupported os {vmOs}"); } } - public static VirtualMachineScaleSetExtensionData AzSecExtension(string region) { - return new VirtualMachineScaleSetExtensionData { - Name = "AzureSecurityLinuxAgent", + public static VMExtenionWrapper AzSecExtension(AzureLocation region) { + return new VMExtenionWrapper { + Location = region, Publisher = "Microsoft.Azure.Security.Monitoring", TypePropertiesType = "AzureSecurityLinuxAgent", TypeHandlerVersion = "2.0", AutoUpgradeMinorVersion = true, - Settings = new BinaryData(new { EnableGenevaUpload = true, EnableAutoConfig = true }) + Settings = new BinaryData(JsonSerializer.Serialize(new { EnableGenevaUpload = true, EnableAutoConfig = true }, _extensionSerializerOptions)) }; } - public static VirtualMachineScaleSetExtensionData AzMonExtension(string region, AzureMonitorExtensionConfig azureMonitor) { + public static VMExtenionWrapper AzMonExtension(AzureLocation region, AzureMonitorExtensionConfig azureMonitor) { var authId = azureMonitor.MonitoringGCSAuthId; var configVersion = azureMonitor.ConfigVersion; var moniker = azureMonitor.Moniker; @@ -140,15 +150,15 @@ public static VirtualMachineScaleSetExtensionData AzMonExtension(string region, var account = azureMonitor.MonitoringGCSAccount; var authIdType = azureMonitor.MonitoringGCSAuthIdType; - return new VirtualMachineScaleSetExtensionData { - Name = "AzureMonitorLinuxAgent", + return new VMExtenionWrapper { + Location = region, Publisher = "Microsoft.Azure.Monitor", TypePropertiesType = "AzureMonitorLinuxAgent", AutoUpgradeMinorVersion = true, TypeHandlerVersion = "1.0", - Settings = new BinaryData(new { GCS_AUTO_CONFIG = true }), + Settings = new BinaryData(JsonSerializer.Serialize(new { GCS_AUTO_CONFIG = true }, _extensionSerializerOptions)), ProtectedSettings = - new BinaryData( + new BinaryData(JsonSerializer.Serialize( new { ConfigVersion = configVersion, Moniker = moniker, @@ -158,15 +168,13 @@ public static VirtualMachineScaleSetExtensionData AzMonExtension(string region, MonitoringGCSRegion = region, MonitoringGCSAuthId = authId, MonitoringGCSAuthIdType = authIdType, - }) + }, _extensionSerializerOptions)) }; } - - - public static VirtualMachineScaleSetExtensionData GenevaExtension(string region) { - return new VirtualMachineScaleSetExtensionData { - Name = "Microsoft.Azure.Geneva.GenevaMonitoring", + public static VMExtenionWrapper GenevaExtension(AzureLocation region) { + return new VMExtenionWrapper { + Location = region, Publisher = "Microsoft.Azure.Geneva", TypePropertiesType = "GenevaMonitoring", TypeHandlerVersion = "2.0", @@ -175,12 +183,12 @@ public static VirtualMachineScaleSetExtensionData GenevaExtension(string region) }; } - public static VirtualMachineScaleSetExtensionData? DependencyExtension(string region, Os vmOs) { + public static VMExtenionWrapper? DependencyExtension(AzureLocation region, Os vmOs) { if (vmOs == Os.Windows) { - return new VirtualMachineScaleSetExtensionData { + return new VMExtenionWrapper { + Location = region, AutoUpgradeMinorVersion = true, - Name = "DependencyAgentWindows", Publisher = "Microsoft.Azure.Monitoring.DependencyAgent", TypePropertiesType = "DependencyAgentWindows", TypeHandlerVersion = "9.5" @@ -239,8 +247,8 @@ public static VirtualMachineScaleSetExtensionData GenevaExtension(string region) } public async Async.Task UpdateManagedScripts() { - var instanceSpecificSetupSas = _containers.GetContainerSasUrl(new Container("instance-specific-setup"), StorageType.Config, BlobContainerSasPermissions.List | BlobContainerSasPermissions.Read); - var toolsSas = _containers.GetContainerSasUrl(new Container("tools"), StorageType.Config, BlobContainerSasPermissions.List | BlobContainerSasPermissions.Read); + var instanceSpecificSetupSas = await _containers.GetContainerSasUrl(new Container("instance-specific-setup"), StorageType.Config, BlobContainerSasPermissions.List | BlobContainerSasPermissions.Read); + var toolsSas = await _containers.GetContainerSasUrl(new Container("tools"), StorageType.Config, BlobContainerSasPermissions.List | BlobContainerSasPermissions.Read); string[] commands = { $"azcopy sync '{instanceSpecificSetupSas}' instance-specific-setup", @@ -251,8 +259,7 @@ public async Async.Task UpdateManagedScripts() { await _containers.SaveBlob(new Container("vm-scripts"), "managed.sh", string.Join("\n", commands) + "\n", StorageType.Config); } - - public async Async.Task AgentConfig(string region, Os vmOs, AgentMode mode, List? urls = null, bool withSas = false) { + public async Async.Task AgentConfig(AzureLocation region, Os vmOs, AgentMode mode, List? urls = null, bool withSas = false) { await UpdateManagedScripts(); var urlsUpdated = urls ?? new(); @@ -267,17 +274,18 @@ public async Async.Task AgentConfig(string urlsUpdated.Add(toolsSetup); urlsUpdated.Add(toolsOneFuzz); - var toExecuteCmd = $"powershell -ExecutionPolicy Unrestricted -File win64/setup.ps1 -mode {mode}"; + var toExecuteCmd = $"powershell -ExecutionPolicy Unrestricted -File win64/setup.ps1 -mode {mode.ToString().ToLowerInvariant()}"; - var extension = new VirtualMachineScaleSetExtensionData { + var extension = new VMExtenionWrapper { Name = "CustomScriptExtension", TypePropertiesType = "CustomScriptExtension", Publisher = "Microsoft.Compute", + Location = region, ForceUpdateTag = Guid.NewGuid().ToString(), TypeHandlerVersion = "1.9", AutoUpgradeMinorVersion = true, - Settings = new BinaryData(new { commandToExecute = toExecuteCmd, fileUrls = urlsUpdated }), - ProtectedSettings = new BinaryData(new { managedIdentity = new Dictionary() }) + Settings = new BinaryData(JsonSerializer.Serialize(new { commandToExecute = toExecuteCmd, fileUris = urlsUpdated }, _extensionSerializerOptions)), + ProtectedSettings = new BinaryData(JsonSerializer.Serialize(new { managedIdentity = new Dictionary() }, _extensionSerializerOptions)) }; return extension; } else if (vmOs == Os.Linux) { @@ -290,17 +298,20 @@ public async Async.Task AgentConfig(string urlsUpdated.Add(toolsAzCopy); urlsUpdated.Add(toolsSetup); - var toExecuteCmd = $"sh setup.sh {mode}"; + var toExecuteCmd = $"sh setup.sh {mode.ToString().ToLowerInvariant()}"; + var extensionSettings = JsonSerializer.Serialize(new { CommandToExecute = toExecuteCmd, FileUris = urlsUpdated }, _extensionSerializerOptions); + var protectedExtensionSettings = JsonSerializer.Serialize(new { ManagedIdentity = new Dictionary() }, _extensionSerializerOptions); - var extension = new VirtualMachineScaleSetExtensionData { + var extension = new VMExtenionWrapper { Name = "CustomScript", + Publisher = "Microsoft.Azure.Extensions", TypePropertiesType = "CustomScript", - Publisher = "Microsoft.Azure.Extension", - ForceUpdateTag = Guid.NewGuid().ToString(), TypeHandlerVersion = "2.1", + Location = region, + ForceUpdateTag = Guid.NewGuid().ToString(), AutoUpgradeMinorVersion = true, - Settings = new BinaryData(new { CommandToExecute = toExecuteCmd, FileUrls = urlsUpdated }), - ProtectedSettings = new BinaryData(new { ManagedIdentity = new Dictionary() }) + Settings = new BinaryData(extensionSettings), + ProtectedSettings = new BinaryData(protectedExtensionSettings) }; return extension; } @@ -308,28 +319,31 @@ public async Async.Task AgentConfig(string throw new NotImplementedException($"unsupported OS: {vmOs}"); } - public async Async.Task MonitorExtension(string region, Os vmOs) { + public async Async.Task MonitorExtension(AzureLocation region, Os vmOs) { var settings = await _logAnalytics.GetMonitorSettings(); - + var extensionSettings = JsonSerializer.Serialize(new { WorkspaceId = settings.Id }, _extensionSerializerOptions); + var protectedExtensionSettings = JsonSerializer.Serialize(new { WorkspaceKey = settings.Key }, _extensionSerializerOptions); if (vmOs == Os.Windows) { - return new VirtualMachineScaleSetExtensionData { + return new VMExtenionWrapper { + Location = region, Name = "OMSExtension", TypePropertiesType = "MicrosoftMonitoringAgent", Publisher = "Microsoft.EnterpriseCloud.Monitoring", TypeHandlerVersion = "1.0", AutoUpgradeMinorVersion = true, - Settings = new BinaryData(new { WorkSpaceId = settings.Id }), - ProtectedSettings = new BinaryData(new { WorkspaceKey = settings.Key }) + Settings = new BinaryData(extensionSettings), + ProtectedSettings = new BinaryData(protectedExtensionSettings) }; } else if (vmOs == Os.Linux) { - return new VirtualMachineScaleSetExtensionData { + return new VMExtenionWrapper { + Location = region, Name = "OMSExtension", TypePropertiesType = "OmsAgentForLinux", Publisher = "Microsoft.EnterpriseCloud.Monitoring", TypeHandlerVersion = "1.12", AutoUpgradeMinorVersion = true, - Settings = new BinaryData(new { WorkSpaceId = settings.Id }), - ProtectedSettings = new BinaryData(new { WorkspaceKey = settings.Key }) + Settings = new BinaryData(extensionSettings), + ProtectedSettings = new BinaryData(protectedExtensionSettings) }; } else { throw new NotImplementedException($"unsupported os: {vmOs}"); @@ -346,6 +360,100 @@ public async Async.Task> FuzzExtensio var extensions = await GenericExtensions(scaleset.Region, pool.Os); extensions.Add(fuzzExtension); - return extensions; + return extensions.Select(extension => extension.GetAsVirtualMachineScaleSetExtension()).ToList(); + } + + public async Task> ReproExtensions(AzureLocation region, Os reproOs, Guid reproId, ReproConfig reproConfig, Container? setupContainer) { + // TODO: what about contents of repro.ps1 / repro.sh? + var report = await _context.Reports.GetReport(reproConfig.Container, reproConfig.Path); + report.EnsureNotNull($"invalid report: {reproConfig}"); + report?.InputBlob.EnsureNotNull("unable to perform reproduction without an input blob"); + + var commands = new List(); + if (setupContainer != null) { + var containerSasUrl = await _context.Containers.GetContainerSasUrl( + setupContainer, + StorageType.Corpus, + BlobContainerSasPermissions.Read | BlobContainerSasPermissions.List + ); + commands.Add( + $"azcopy sync '{containerSasUrl}' ./setup" + ); + } + + var urls = new List() + { + (await _context.Containers.GetFileSasUrl( + reproConfig.Container, + reproConfig.Path, + StorageType.Corpus, + BlobSasPermissions.Read + )), + (await _context.Containers.GetFileSasUrl( + report?.InputBlob?.container!, + report?.InputBlob?.Name!, + StorageType.Corpus, + BlobSasPermissions.Read + )) + }; + + List reproFiles; + string taskScript; + string scriptName; + if (reproOs == Os.Windows) { + reproFiles = new List() + { + $"{reproId}/repro.ps1" + }; + taskScript = string.Join("\r\n", commands); + scriptName = "task-setup.ps1"; + } else { + reproFiles = new List() + { + $"{reproId}/repro.sh", + $"{reproId}/repro-stdout.sh" + }; + commands.Add("chmod -R +x setup"); + taskScript = string.Join("\n", commands); + scriptName = "task-setup.sh"; + } + + await _context.Containers.SaveBlob( + new Container("task-configs"), + $"{reproId}/{scriptName}", + taskScript, + StorageType.Config + ); + + foreach (var reproFile in reproFiles) { + urls.AddRange(new List() + { + (await _context.Containers.GetFileSasUrl( + new Container("repro-scripts"), + reproFile, + StorageType.Config, + BlobSasPermissions.Read + )), + (await _context.Containers.GetFileSasUrl( + new Container("task-configs"), + $"{reproId}/{scriptName}", + StorageType.Config, + BlobSasPermissions.Read + )) + }); + } + + var baseExtension = await AgentConfig(region, reproOs, AgentMode.Repro, urls: urls, withSas: true); + var extensions = await GenericExtensions(region, reproOs); + extensions.Add(baseExtension); + + var extensionsDict = new Dictionary(); + foreach (var extension in extensions) { + var (name, data) = extension.GetAsVirtualMachineExtension(); + extensionsDict.Add(name, data); + } + + return extensionsDict; } + } diff --git a/src/ApiService/ApiService/onefuzzlib/ImageOperations.cs b/src/ApiService/ApiService/onefuzzlib/ImageOperations.cs new file mode 100644 index 0000000000..e8b5e7c573 --- /dev/null +++ b/src/ApiService/ApiService/onefuzzlib/ImageOperations.cs @@ -0,0 +1,115 @@ +using System.Threading.Tasks; +using Azure; +using Azure.ResourceManager.Compute; + +namespace Microsoft.OneFuzz.Service; + +public interface IImageOperations { + public Async.Task> GetOs(string region, string image); +} + +public class ImageOperations : IImageOperations { + private IOnefuzzContext _context; + private ILogTracer _logTracer; + + public ImageOperations(ILogTracer logTracer, IOnefuzzContext context) { + _logTracer = logTracer; + _context = context; + } + public async Task> GetOs(string region, string image) { + string? name = null; + try { + var parsed = _context.Creds.ParseResourceId(image); + var _ = !parsed.HasData ? await parsed.GetAsync() : null; + if (string.Equals(parsed.Id.ResourceType, "galleries", StringComparison.OrdinalIgnoreCase)) { + try { + // This is not _exactly_ the same as the python code + // because in C# we don't have access to child_name_1 + var gallery = await _context.Creds.GetResourceGroupResource().GetGalleries().GetAsync( + parsed.Data.Name + ); + + var galleryImage = gallery.Value.GetGalleryImages() + .ToEnumerable() + .Where(galleryImage => string.Equals(galleryImage.Id, parsed.Id, StringComparison.OrdinalIgnoreCase)) + .First(); + + galleryImage = await galleryImage.GetAsync(); + + name = galleryImage.Data?.OSType?.ToString().ToLowerInvariant()!; + + } catch (Exception ex) when ( + ex is RequestFailedException || + ex is NullReferenceException + ) { + return OneFuzzResult.Error( + ErrorCode.INVALID_IMAGE, + ex.ToString() + ); + } + } else { + try { + name = (await _context.Creds.GetResourceGroupResource().GetImages().GetAsync( + parsed.Data.Name + )).Value.Data.StorageProfile.OSDisk.OSType.ToString().ToLowerInvariant(); + } catch (Exception ex) when ( + ex is RequestFailedException || + ex is NullReferenceException + ) { + return OneFuzzResult.Error( + ErrorCode.INVALID_IMAGE, + ex.ToString() + ); + } + } + } catch (FormatException) { + var imageParts = image.Split(":"); + + // The python code would throw if more than 4 parts are found in the split + System.Diagnostics.Trace.Assert(imageParts.Length == 4, $"Expected 4 ':' separated parts in {image}"); + + var publisher = imageParts[0]; + var offer = imageParts[1]; + var sku = imageParts[2]; + var version = imageParts[3]; + + try { + var subscription = await _context.Creds.ArmClient.GetDefaultSubscriptionAsync(); + if (string.Equals(version, "latest", StringComparison.Ordinal)) { + version = (await subscription.GetVirtualMachineImagesAsync( + region, + publisher, + offer, + sku, + top: 1 + ).FirstAsync()).Name; + } + + name = (await subscription.GetVirtualMachineImageAsync( + region, + publisher, + offer, + sku + , version + )).Value.OSDiskImageOperatingSystem.ToString().ToLower(); + } catch (RequestFailedException ex) { + return OneFuzzResult.Error( + ErrorCode.INVALID_IMAGE, + ex.ToString() + ); + } + } + + if (name != null) { + name = string.Concat(name[0].ToString().ToUpper(), name.AsSpan(1)); + if (Enum.TryParse(name, out Os os)) { + return OneFuzzResult.Ok(os); + } + } + + return OneFuzzResult.Error( + ErrorCode.INVALID_IMAGE, + $"Unexpected image os type: {name}" + ); + } +} diff --git a/src/ApiService/ApiService/onefuzzlib/OnefuzzContext.cs b/src/ApiService/ApiService/onefuzzlib/OnefuzzContext.cs index d4be961854..6d183ff4bb 100644 --- a/src/ApiService/ApiService/onefuzzlib/OnefuzzContext.cs +++ b/src/ApiService/ApiService/onefuzzlib/OnefuzzContext.cs @@ -38,6 +38,7 @@ public interface IOnefuzzContext { IRequestHandling RequestHandling { get; } INsgOperations NsgOperations { get; } ISubnet Subnet { get; } + IImageOperations ImageOperations { get; } } public class OnefuzzContext : IOnefuzzContext { @@ -81,4 +82,5 @@ public OnefuzzContext(IServiceProvider serviceProvider) { public IRequestHandling RequestHandling => _serviceProvider.GetRequiredService(); public INsgOperations NsgOperations => _serviceProvider.GetRequiredService(); public ISubnet Subnet => _serviceProvider.GetRequiredService(); + public IImageOperations ImageOperations => _serviceProvider.GetRequiredService(); } diff --git a/src/ApiService/ApiService/onefuzzlib/Reports.cs b/src/ApiService/ApiService/onefuzzlib/Reports.cs index 917ed3979c..d31f75f7ec 100644 --- a/src/ApiService/ApiService/onefuzzlib/Reports.cs +++ b/src/ApiService/ApiService/onefuzzlib/Reports.cs @@ -1,10 +1,12 @@ using System.Text.Json; +using System.Threading.Tasks; using Microsoft.OneFuzz.Service.OneFuzzLib.Orm; namespace Microsoft.OneFuzz.Service; public interface IReports { public Async.Task GetReportOrRegression(Container container, string fileName, bool expectReports = false, params string[] args); + public Async.Task GetReport(Container container, string fileName); } public class Reports : IReports { @@ -15,6 +17,15 @@ public Reports(ILogTracer log, IContainers containers) { _containers = containers; } + public async Task GetReport(Container container, string fileName) { + var result = await GetReportOrRegression(container, fileName); + if (result != null && result is Report) { + return result as Report; + } + + return null; + } + public async Async.Task GetReportOrRegression(Container container, string fileName, bool expectReports = false, params string[] args) { var filePath = String.Join("/", new[] { container.ContainerName, fileName }); if (!fileName.EndsWith(".json", StringComparison.Ordinal)) { diff --git a/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs b/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs new file mode 100644 index 0000000000..4c45fe4b43 --- /dev/null +++ b/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs @@ -0,0 +1,62 @@ +using Azure.Core; +using Azure.ResourceManager.Compute; + +namespace Microsoft.OneFuzz.Service { + public class VMExtenionWrapper { + public AzureLocation? Location { get; init; } + public string? Name { get; init; } + public string? TypePropertiesType { get; init; } + public string? Publisher { get; init; } + public string? TypeHandlerVersion { get; init; } + public string? ForceUpdateTag { get; init; } + public bool? AutoUpgradeMinorVersion { get; init; } + public bool? EnableAutomaticUpgrade { get; init; } + public BinaryData? Settings { get; init; } + public BinaryData? ProtectedSettings { get; init; } + + public (string, VirtualMachineExtensionData) GetAsVirtualMachineExtension() { + if (Location == null) { // EnsureNotNull does not satisfy the nullability checker + throw new ArgumentNullException("Location required for VirtualMachineExtension"); + } + TypePropertiesType.EnsureNotNull("TypePropertiesType required for VirtualMachineExtension"); + Publisher.EnsureNotNull("Publisher required for VirtualMachineExtension"); + TypeHandlerVersion.EnsureNotNull("TypeHandlerVersion required for VirtualMachineExtension"); + AutoUpgradeMinorVersion.EnsureNotNull("AutoUpgradeMinorVersion required for VirtualMachineExtension"); + Settings.EnsureNotNull("Settings required for VirtualMachineExtension"); + ProtectedSettings.EnsureNotNull("ProtectedSettings required for VirtualMachineExtension"); + + return (Name!, new VirtualMachineExtensionData(Location.Value) { + TypePropertiesType = TypePropertiesType, + Publisher = Publisher, + TypeHandlerVersion = TypeHandlerVersion, + AutoUpgradeMinorVersion = AutoUpgradeMinorVersion, + EnableAutomaticUpgrade = EnableAutomaticUpgrade, + ForceUpdateTag = ForceUpdateTag, + Settings = Settings, + ProtectedSettings = ProtectedSettings + }); + } + + public VirtualMachineScaleSetExtensionData GetAsVirtualMachineScaleSetExtension() { + Name.EnsureNotNull("Name required for VirtualMachineScaleSetExtension"); + TypePropertiesType.EnsureNotNull("TypePropertiesType required for VirtualMachineScaleSetExtension"); + Publisher.EnsureNotNull("Publisher required for VirtualMachineScaleSetExtension"); + TypeHandlerVersion.EnsureNotNull("TypeHandlerVersion required for VirtualMachineScaleSetExtension"); + AutoUpgradeMinorVersion.EnsureNotNull("AutoUpgradeMinorVersion required for VirtualMachineScaleSetExtension"); + Settings.EnsureNotNull("Settings required for VirtualMachineScaleSetExtension"); + ProtectedSettings.EnsureNotNull("ProtectedSettings required for VirtualMachineScaleSetExtension"); + return new VirtualMachineScaleSetExtensionData() { + Name = Name, + TypePropertiesType = TypePropertiesType, + Publisher = Publisher, + TypeHandlerVersion = TypeHandlerVersion, + AutoUpgradeMinorVersion = AutoUpgradeMinorVersion, + EnableAutomaticUpgrade = EnableAutomaticUpgrade, + ForceUpdateTag = ForceUpdateTag, + Settings = Settings, + ProtectedSettings = ProtectedSettings + }; + } + } + +} diff --git a/src/ApiService/ApiService/onefuzzlib/orm/Orm.cs b/src/ApiService/ApiService/onefuzzlib/orm/Orm.cs index 5063a346e8..e1cf7d366c 100644 --- a/src/ApiService/ApiService/onefuzzlib/orm/Orm.cs +++ b/src/ApiService/ApiService/onefuzzlib/orm/Orm.cs @@ -211,6 +211,9 @@ public StatefulOrm(ILogTracer logTracer, IOnefuzzContext context) : base(logTrac _logTracer.Info($"processing state update: {typeof(T)} - PartitionKey {_partitionKeyGetter?.Value()} {_rowKeyGetter?.Value()} - %s"); return await func(entity); } + else { + _logTracer.Info($"State function for state: '{state}' not found on type {typeof(T)}"); + } return null; } diff --git a/src/ApiService/IntegrationTests/Fakes/TestContext.cs b/src/ApiService/IntegrationTests/Fakes/TestContext.cs index cfec64971f..80535c4c38 100644 --- a/src/ApiService/IntegrationTests/Fakes/TestContext.cs +++ b/src/ApiService/IntegrationTests/Fakes/TestContext.cs @@ -109,4 +109,6 @@ public Async.Task InsertAll(params EntityBase[] objs) public INsgOperations NsgOperations => throw new NotImplementedException(); public ISubnet Subnet => throw new NotImplementedException(); + + public IImageOperations ImageOperations => throw new NotImplementedException(); } From 8e6a18ffa9704a6e7aa8cffc977a00170365f5cb Mon Sep 17 00:00:00 2001 From: Teo Voinea Date: Thu, 4 Aug 2022 18:14:04 +0000 Subject: [PATCH 2/3] Small cleanup --- .../ApiService/onefuzzlib/Extension.cs | 22 ++++++++++++------- .../ApiService/onefuzzlib/ImageOperations.cs | 4 ++-- .../onefuzzlib/VmExtensionWrapper.cs | 2 +- .../ApiService/onefuzzlib/orm/Orm.cs | 3 +-- 4 files changed, 18 insertions(+), 13 deletions(-) diff --git a/src/ApiService/ApiService/onefuzzlib/Extension.cs b/src/ApiService/ApiService/onefuzzlib/Extension.cs index 431f608e1a..6a4d72ed4f 100644 --- a/src/ApiService/ApiService/onefuzzlib/Extension.cs +++ b/src/ApiService/ApiService/onefuzzlib/Extension.cs @@ -89,6 +89,7 @@ public static VMExtenionWrapper KeyVaultExtension(AzureLocation region, Keyvault if (vmOs == Os.Windows) { return new VMExtenionWrapper { Location = region, + Name = "KVVMExtensionForWindows", Publisher = "Microsoft.Azure.KeyVault", TypePropertiesType = "KeyVaultForWindows", TypeHandlerVersion = "1.0", @@ -111,6 +112,7 @@ public static VMExtenionWrapper KeyVaultExtension(AzureLocation region, Keyvault return new VMExtenionWrapper { Location = region, + Name = "KVVMExtensionForLinux", Publisher = "Microsoft.Azure.KeyVault", TypePropertiesType = "KeyVaultForLinux", TypeHandlerVersion = "2.0", @@ -132,6 +134,7 @@ public static VMExtenionWrapper KeyVaultExtension(AzureLocation region, Keyvault public static VMExtenionWrapper AzSecExtension(AzureLocation region) { return new VMExtenionWrapper { Location = region, + Name = "AzureSecurityLinuxAgent", Publisher = "Microsoft.Azure.Security.Monitoring", TypePropertiesType = "AzureSecurityLinuxAgent", TypeHandlerVersion = "2.0", @@ -152,6 +155,7 @@ public static VMExtenionWrapper AzMonExtension(AzureLocation region, AzureMonito return new VMExtenionWrapper { Location = region, + Name = "AzureMonitorLinuxAgent", Publisher = "Microsoft.Azure.Monitor", TypePropertiesType = "AzureMonitorLinuxAgent", AutoUpgradeMinorVersion = true, @@ -175,6 +179,7 @@ public static VMExtenionWrapper AzMonExtension(AzureLocation region, AzureMonito public static VMExtenionWrapper GenevaExtension(AzureLocation region) { return new VMExtenionWrapper { Location = region, + Name = "Microsoft.Azure.Geneva.GenevaMonitoring", Publisher = "Microsoft.Azure.Geneva", TypePropertiesType = "GenevaMonitoring", TypeHandlerVersion = "2.0", @@ -188,6 +193,7 @@ public static VMExtenionWrapper GenevaExtension(AzureLocation region) { if (vmOs == Os.Windows) { return new VMExtenionWrapper { Location = region, + Name = "DependencyAgentWindows", AutoUpgradeMinorVersion = true, Publisher = "Microsoft.Azure.Monitoring.DependencyAgent", TypePropertiesType = "DependencyAgentWindows", @@ -383,18 +389,18 @@ public async Task> ReproExtensio var urls = new List() { - (await _context.Containers.GetFileSasUrl( + await _context.Containers.GetFileSasUrl( reproConfig.Container, reproConfig.Path, StorageType.Corpus, BlobSasPermissions.Read - )), - (await _context.Containers.GetFileSasUrl( + ), + await _context.Containers.GetFileSasUrl( report?.InputBlob?.container!, report?.InputBlob?.Name!, StorageType.Corpus, BlobSasPermissions.Read - )) + ) }; List reproFiles; @@ -428,18 +434,18 @@ await _context.Containers.SaveBlob( foreach (var reproFile in reproFiles) { urls.AddRange(new List() { - (await _context.Containers.GetFileSasUrl( + await _context.Containers.GetFileSasUrl( new Container("repro-scripts"), reproFile, StorageType.Config, BlobSasPermissions.Read - )), - (await _context.Containers.GetFileSasUrl( + ), + await _context.Containers.GetFileSasUrl( new Container("task-configs"), $"{reproId}/{scriptName}", StorageType.Config, BlobSasPermissions.Read - )) + ) }); } diff --git a/src/ApiService/ApiService/onefuzzlib/ImageOperations.cs b/src/ApiService/ApiService/onefuzzlib/ImageOperations.cs index e8b5e7c573..67c2fd62f7 100644 --- a/src/ApiService/ApiService/onefuzzlib/ImageOperations.cs +++ b/src/ApiService/ApiService/onefuzzlib/ImageOperations.cs @@ -1,4 +1,4 @@ -using System.Threading.Tasks; +using System.Threading.Tasks; using Azure; using Azure.ResourceManager.Compute; @@ -20,7 +20,7 @@ public async Task> GetOs(string region, string image) { string? name = null; try { var parsed = _context.Creds.ParseResourceId(image); - var _ = !parsed.HasData ? await parsed.GetAsync() : null; + parsed = await _context.Creds.GetData(parsed); if (string.Equals(parsed.Id.ResourceType, "galleries", StringComparison.OrdinalIgnoreCase)) { try { // This is not _exactly_ the same as the python code diff --git a/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs b/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs index 4c45fe4b43..d013272d48 100644 --- a/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs +++ b/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs @@ -1,4 +1,4 @@ -using Azure.Core; +using Azure.Core; using Azure.ResourceManager.Compute; namespace Microsoft.OneFuzz.Service { diff --git a/src/ApiService/ApiService/onefuzzlib/orm/Orm.cs b/src/ApiService/ApiService/onefuzzlib/orm/Orm.cs index e1cf7d366c..ebad8d8233 100644 --- a/src/ApiService/ApiService/onefuzzlib/orm/Orm.cs +++ b/src/ApiService/ApiService/onefuzzlib/orm/Orm.cs @@ -210,8 +210,7 @@ public StatefulOrm(ILogTracer logTracer, IOnefuzzContext context) : base(logTrac if (func != null) { _logTracer.Info($"processing state update: {typeof(T)} - PartitionKey {_partitionKeyGetter?.Value()} {_rowKeyGetter?.Value()} - %s"); return await func(entity); - } - else { + } else { _logTracer.Info($"State function for state: '{state}' not found on type {typeof(T)}"); } return null; From 3eacd02ec2d15ad273d2ba222f47ac13b9de361c Mon Sep 17 00:00:00 2001 From: Teo Voinea Date: Thu, 4 Aug 2022 18:59:17 +0000 Subject: [PATCH 3/3] Use context, fix typo --- .../ApiService/onefuzzlib/Extension.cs | 87 ++++++++----------- .../onefuzzlib/VmExtensionWrapper.cs | 2 +- 2 files changed, 38 insertions(+), 51 deletions(-) diff --git a/src/ApiService/ApiService/onefuzzlib/Extension.cs b/src/ApiService/ApiService/onefuzzlib/Extension.cs index 6a4d72ed4f..7178d1cf35 100644 --- a/src/ApiService/ApiService/onefuzzlib/Extension.cs +++ b/src/ApiService/ApiService/onefuzzlib/Extension.cs @@ -14,40 +14,27 @@ public interface IExtensions { } public class Extensions : IExtensions { - IServiceConfig _serviceConfig; - ICreds _creds; - IQueue _queue; - IContainers _containers; - IConfigOperations _instanceConfigOps; - ILogAnalytics _logAnalytics; - IOnefuzzContext _context; private static readonly JsonSerializerOptions _extensionSerializerOptions = new JsonSerializerOptions { PropertyNamingPolicy = JsonNamingPolicy.CamelCase }; - public Extensions(IServiceConfig config, ICreds creds, IQueue queue, IContainers containers, IConfigOperations instanceConfigOps, ILogAnalytics logAnalytics, IOnefuzzContext context) { - _serviceConfig = config; - _creds = creds; - _queue = queue; - _containers = containers; - _instanceConfigOps = instanceConfigOps; - _logAnalytics = logAnalytics; + public Extensions(IOnefuzzContext context) { _context = context; } public async Async.Task ConfigUrl(Container container, string fileName, bool withSas) { if (withSas) - return await _containers.GetFileSasUrl(container, fileName, StorageType.Config, BlobSasPermissions.Read); + return await _context.Containers.GetFileSasUrl(container, fileName, StorageType.Config, BlobSasPermissions.Read); else - return await _containers.GetFileUrl(container, fileName, StorageType.Config); + return await _context.Containers.GetFileUrl(container, fileName, StorageType.Config); } - public async Async.Task> GenericExtensions(AzureLocation region, Os vmOs) { - var extensions = new List(); + public async Async.Task> GenericExtensions(AzureLocation region, Os vmOs) { + var extensions = new List(); - var instanceConfig = await _instanceConfigOps.Fetch(); + var instanceConfig = await _context.ConfigOperations.Fetch(); extensions.Add(await MonitorExtension(region, vmOs)); var depenency = DependencyExtension(region, vmOs); @@ -81,13 +68,13 @@ public async Async.Task> GenericExtensions(AzureLocatio return extensions; } - public static VMExtenionWrapper KeyVaultExtension(AzureLocation region, KeyvaultExtensionConfig keyVault, Os vmOs) { + public static VMExtensionWrapper KeyVaultExtension(AzureLocation region, KeyvaultExtensionConfig keyVault, Os vmOs) { var keyVaultName = keyVault.KeyVaultName; var certName = keyVault.CertName; var uri = keyVaultName + certName; if (vmOs == Os.Windows) { - return new VMExtenionWrapper { + return new VMExtensionWrapper { Location = region, Name = "KVVMExtensionForWindows", Publisher = "Microsoft.Azure.KeyVault", @@ -110,7 +97,7 @@ public static VMExtenionWrapper KeyVaultExtension(AzureLocation region, Keyvault var extensionStore = keyVault.ExtensionStore; var certLocation = certPath + extensionStore; - return new VMExtenionWrapper { + return new VMExtensionWrapper { Location = region, Name = "KVVMExtensionForLinux", Publisher = "Microsoft.Azure.KeyVault", @@ -131,8 +118,8 @@ public static VMExtenionWrapper KeyVaultExtension(AzureLocation region, Keyvault } } - public static VMExtenionWrapper AzSecExtension(AzureLocation region) { - return new VMExtenionWrapper { + public static VMExtensionWrapper AzSecExtension(AzureLocation region) { + return new VMExtensionWrapper { Location = region, Name = "AzureSecurityLinuxAgent", Publisher = "Microsoft.Azure.Security.Monitoring", @@ -144,7 +131,7 @@ public static VMExtenionWrapper AzSecExtension(AzureLocation region) { } - public static VMExtenionWrapper AzMonExtension(AzureLocation region, AzureMonitorExtensionConfig azureMonitor) { + public static VMExtensionWrapper AzMonExtension(AzureLocation region, AzureMonitorExtensionConfig azureMonitor) { var authId = azureMonitor.MonitoringGCSAuthId; var configVersion = azureMonitor.ConfigVersion; var moniker = azureMonitor.Moniker; @@ -153,7 +140,7 @@ public static VMExtenionWrapper AzMonExtension(AzureLocation region, AzureMonito var account = azureMonitor.MonitoringGCSAccount; var authIdType = azureMonitor.MonitoringGCSAuthIdType; - return new VMExtenionWrapper { + return new VMExtensionWrapper { Location = region, Name = "AzureMonitorLinuxAgent", Publisher = "Microsoft.Azure.Monitor", @@ -176,8 +163,8 @@ public static VMExtenionWrapper AzMonExtension(AzureLocation region, AzureMonito }; } - public static VMExtenionWrapper GenevaExtension(AzureLocation region) { - return new VMExtenionWrapper { + public static VMExtensionWrapper GenevaExtension(AzureLocation region) { + return new VMExtensionWrapper { Location = region, Name = "Microsoft.Azure.Geneva.GenevaMonitoring", Publisher = "Microsoft.Azure.Geneva", @@ -188,10 +175,10 @@ public static VMExtenionWrapper GenevaExtension(AzureLocation region) { }; } - public static VMExtenionWrapper? DependencyExtension(AzureLocation region, Os vmOs) { + public static VMExtensionWrapper? DependencyExtension(AzureLocation region, Os vmOs) { if (vmOs == Os.Windows) { - return new VMExtenionWrapper { + return new VMExtensionWrapper { Location = region, Name = "DependencyAgentWindows", AutoUpgradeMinorVersion = true, @@ -216,22 +203,22 @@ public static VMExtenionWrapper GenevaExtension(AzureLocation region) { public async Async.Task BuildPoolConfig(Pool pool) { - var instanceId = await _containers.GetInstanceId(); + var instanceId = await _context.Containers.GetInstanceId(); - var queueSas = await _queue.GetQueueSas("node-heartbeat", StorageType.Config, QueueSasPermissions.Add); + var queueSas = await _context.Queue.GetQueueSas("node-heartbeat", StorageType.Config, QueueSasPermissions.Add); var config = new AgentConfig( ClientCredentials: null, - OneFuzzUrl: _creds.GetInstanceUrl(), + OneFuzzUrl: _context.Creds.GetInstanceUrl(), PoolName: pool.Name, HeartbeatQueue: queueSas, - InstanceTelemetryKey: _serviceConfig.ApplicationInsightsInstrumentationKey, - MicrosoftTelemetryKey: _serviceConfig.OneFuzzTelemetry, - MultiTenantDomain: _serviceConfig.MultiTenantDomain, + InstanceTelemetryKey: _context.ServiceConfiguration.ApplicationInsightsInstrumentationKey, + MicrosoftTelemetryKey: _context.ServiceConfiguration.OneFuzzTelemetry, + MultiTenantDomain: _context.ServiceConfiguration.MultiTenantDomain, InstanceId: instanceId ); var fileName = $"{pool.Name}/config.json"; - await _containers.SaveBlob(new Container("vm-scripts"), fileName, (JsonSerializer.Serialize(config, EntityConverter.GetJsonSerializerOptions())), StorageType.Config); + await _context.Containers.SaveBlob(new Container("vm-scripts"), fileName, (JsonSerializer.Serialize(config, EntityConverter.GetJsonSerializerOptions())), StorageType.Config); return await ConfigUrl(new Container("vm-scripts"), fileName, false); } @@ -248,24 +235,24 @@ public static VMExtenionWrapper GenevaExtension(AzureLocation region) { commands.Add($"Set-Content -Path {sshPath} -Value \"{sshKey}\""); } - await _containers.SaveBlob(new Container("vm-scripts"), fileName, string.Join(sep, commands) + sep, StorageType.Config); - return await _containers.GetFileUrl(new Container("vm-scripts"), fileName, StorageType.Config); + await _context.Containers.SaveBlob(new Container("vm-scripts"), fileName, string.Join(sep, commands) + sep, StorageType.Config); + return await _context.Containers.GetFileUrl(new Container("vm-scripts"), fileName, StorageType.Config); } public async Async.Task UpdateManagedScripts() { - var instanceSpecificSetupSas = await _containers.GetContainerSasUrl(new Container("instance-specific-setup"), StorageType.Config, BlobContainerSasPermissions.List | BlobContainerSasPermissions.Read); - var toolsSas = await _containers.GetContainerSasUrl(new Container("tools"), StorageType.Config, BlobContainerSasPermissions.List | BlobContainerSasPermissions.Read); + var instanceSpecificSetupSas = await _context.Containers.GetContainerSasUrl(new Container("instance-specific-setup"), StorageType.Config, BlobContainerSasPermissions.List | BlobContainerSasPermissions.Read); + var toolsSas = await _context.Containers.GetContainerSasUrl(new Container("tools"), StorageType.Config, BlobContainerSasPermissions.List | BlobContainerSasPermissions.Read); string[] commands = { $"azcopy sync '{instanceSpecificSetupSas}' instance-specific-setup", $"azcopy sync '{toolsSas}' tools" }; - await _containers.SaveBlob(new Container("vm-scripts"), "managed.ps1", string.Join("\r\n", commands) + "\r\n", StorageType.Config); - await _containers.SaveBlob(new Container("vm-scripts"), "managed.sh", string.Join("\n", commands) + "\n", StorageType.Config); + await _context.Containers.SaveBlob(new Container("vm-scripts"), "managed.ps1", string.Join("\r\n", commands) + "\r\n", StorageType.Config); + await _context.Containers.SaveBlob(new Container("vm-scripts"), "managed.sh", string.Join("\n", commands) + "\n", StorageType.Config); } - public async Async.Task AgentConfig(AzureLocation region, Os vmOs, AgentMode mode, List? urls = null, bool withSas = false) { + public async Async.Task AgentConfig(AzureLocation region, Os vmOs, AgentMode mode, List? urls = null, bool withSas = false) { await UpdateManagedScripts(); var urlsUpdated = urls ?? new(); @@ -282,7 +269,7 @@ public async Async.Task AgentConfig(AzureLocation region, Os var toExecuteCmd = $"powershell -ExecutionPolicy Unrestricted -File win64/setup.ps1 -mode {mode.ToString().ToLowerInvariant()}"; - var extension = new VMExtenionWrapper { + var extension = new VMExtensionWrapper { Name = "CustomScriptExtension", TypePropertiesType = "CustomScriptExtension", Publisher = "Microsoft.Compute", @@ -308,7 +295,7 @@ public async Async.Task AgentConfig(AzureLocation region, Os var extensionSettings = JsonSerializer.Serialize(new { CommandToExecute = toExecuteCmd, FileUris = urlsUpdated }, _extensionSerializerOptions); var protectedExtensionSettings = JsonSerializer.Serialize(new { ManagedIdentity = new Dictionary() }, _extensionSerializerOptions); - var extension = new VMExtenionWrapper { + var extension = new VMExtensionWrapper { Name = "CustomScript", Publisher = "Microsoft.Azure.Extensions", TypePropertiesType = "CustomScript", @@ -325,12 +312,12 @@ public async Async.Task AgentConfig(AzureLocation region, Os throw new NotImplementedException($"unsupported OS: {vmOs}"); } - public async Async.Task MonitorExtension(AzureLocation region, Os vmOs) { - var settings = await _logAnalytics.GetMonitorSettings(); + public async Async.Task MonitorExtension(AzureLocation region, Os vmOs) { + var settings = await _context.LogAnalytics.GetMonitorSettings(); var extensionSettings = JsonSerializer.Serialize(new { WorkspaceId = settings.Id }, _extensionSerializerOptions); var protectedExtensionSettings = JsonSerializer.Serialize(new { WorkspaceKey = settings.Key }, _extensionSerializerOptions); if (vmOs == Os.Windows) { - return new VMExtenionWrapper { + return new VMExtensionWrapper { Location = region, Name = "OMSExtension", TypePropertiesType = "MicrosoftMonitoringAgent", @@ -341,7 +328,7 @@ public async Async.Task MonitorExtension(AzureLocation region ProtectedSettings = new BinaryData(protectedExtensionSettings) }; } else if (vmOs == Os.Linux) { - return new VMExtenionWrapper { + return new VMExtensionWrapper { Location = region, Name = "OMSExtension", TypePropertiesType = "OmsAgentForLinux", diff --git a/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs b/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs index d013272d48..2113378e96 100644 --- a/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs +++ b/src/ApiService/ApiService/onefuzzlib/VmExtensionWrapper.cs @@ -2,7 +2,7 @@ using Azure.ResourceManager.Compute; namespace Microsoft.OneFuzz.Service { - public class VMExtenionWrapper { + public class VMExtensionWrapper { public AzureLocation? Location { get; init; } public string? Name { get; init; } public string? TypePropertiesType { get; init; }