From c980191496e35b4e6dc82208baed675a49aa7792 Mon Sep 17 00:00:00 2001 From: George Pollard Date: Mon, 25 Jul 2022 22:42:36 +0000 Subject: [PATCH 01/14] Start implementing `scaleset` function --- .../ApiService/Functions/Scaleset.cs | 97 +++++++++++ .../ApiService/OneFuzzTypes/Model.cs | 5 +- .../ApiService/OneFuzzTypes/Requests.cs | 34 +++- .../ApiService/OneFuzzTypes/Responses.cs | 39 ++++- .../onefuzzlib/ScalesetOperations.cs | 160 ++++++++++++------ .../ApiService/onefuzzlib/VmssOperations.cs | 37 +--- 6 files changed, 281 insertions(+), 91 deletions(-) create mode 100644 src/ApiService/ApiService/Functions/Scaleset.cs diff --git a/src/ApiService/ApiService/Functions/Scaleset.cs b/src/ApiService/ApiService/Functions/Scaleset.cs new file mode 100644 index 0000000000..cc9fba97fc --- /dev/null +++ b/src/ApiService/ApiService/Functions/Scaleset.cs @@ -0,0 +1,97 @@ +using System.Threading.Tasks; +using Microsoft.Azure.Functions.Worker; +using Microsoft.Azure.Functions.Worker.Http; + +namespace Microsoft.OneFuzz.Service.Functions; + +public class Scaleset { + private readonly ILogTracer _log; + private readonly IEndpointAuthorization _auth; + private readonly IOnefuzzContext _context; + + public Scaleset(ILogTracer log, IEndpointAuthorization auth, IOnefuzzContext context) { + _log = log; + _auth = auth; + _context = context; + } + + [Function("Scaleset")] + public Async.Task Run([HttpTrigger(AuthorizationLevel.Anonymous, "GET", "PATCH", "POST", "DELETE")] HttpRequestData req) { + return _auth.CallIfUser(req, r => r.Method switch { + "GET" => Get(r), + "PATCH" => Patch(r), + "POST" => Post(r), + "DELETE" => Delete(r), + _ => throw new InvalidOperationException("Unsupported HTTP method"), + }); + } + + private async Task Delete(HttpRequestData req) { + var request = await RequestHandling.ParseRequest(req); + if (!request.IsOk) { + return await _context.RequestHandling.NotOk(req, request.ErrorV, "ScalesetDelete"); + } + + var answer = await _auth.CheckRequireAdmins(req); + if (!answer.IsOk) { + return await _context.RequestHandling.NotOk(req, answer.ErrorV, "ScalesetDelete"); + } + + var scalesetResult = await _context.ScalesetOperations.GetById(request.OkV.ScalesetId); + if (!scalesetResult.IsOk) { + return await _context.RequestHandling.NotOk(req, scalesetResult.ErrorV, "ScalesetDelete"); + } + + var scaleset = scalesetResult.OkV; + await _context.ScalesetOperations.SetShutdown(scaleset, request.OkV.Now); + return await RequestHandling.Ok(req, true); + } + + private async Task Post(HttpRequestData req) { + var request = await RequestHandling.ParseRequest(req); + if (!request.IsOk) { + return await _context.RequestHandling.NotOk(req, request.ErrorV, "ScalesetCreate"); + } + + throw new NotImplementedException(); + } + + private async Task Patch(HttpRequestData req) { + var request = await RequestHandling.ParseRequest(req); + if (!request.IsOk) { + return await _context.RequestHandling.NotOk(req, request.ErrorV, "ScalesetUpdate"); + } + throw new NotImplementedException(); + } + + private async Task Get(HttpRequestData req) { + var request = await RequestHandling.ParseRequest(req); + if (!request.IsOk) { + return await _context.RequestHandling.NotOk(req, request.ErrorV, "ScalesetSearch"); + } + + var search = request.OkV; + if (search.ScalesetId is Guid id) { + var scalesetResult = await _context.ScalesetOperations.GetById(id); + if (!scalesetResult.IsOk) { + return await _context.RequestHandling.NotOk(req, scalesetResult.ErrorV, "ScalesetSearch"); + } + + var scaleset = scalesetResult.OkV; + + var response = ScalesetSearchResponse.ForScaleset(scaleset); + response = response with { Nodes = await _context.ScalesetOperations.GetNodes(scaleset) }; + if (!search.IncludeAuth) { + response = response with { Auth = null }; + } + + return await RequestHandling.Ok(req, response); + } + + var states = search.State ?? Enumerable.Empty(); + var scalesets = await _context.ScalesetOperations.SearchStates(states).ToListAsync(); + // don't return auths during list actions, only 'get' + var result = scalesets.Select(ss => ScalesetSearchResponse.ForScaleset(ss with { Auth = null })); + return await RequestHandling.Ok(req, result); + } +} diff --git a/src/ApiService/ApiService/OneFuzzTypes/Model.cs b/src/ApiService/ApiService/OneFuzzTypes/Model.cs index 5981294205..9c0eb34db0 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Model.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Model.cs @@ -384,15 +384,12 @@ long ScaleInCoolDown ) : EntityBase(); - public record ScalesetNodeState( Guid MachineId, string InstanceId, NodeState? State - ); - public record Scaleset( [PartitionKey] PoolName PoolName, [RowKey] Guid ScalesetId, @@ -406,10 +403,10 @@ public record Scaleset( bool EphemeralOsDisks, bool NeedsConfigUpdate, Error? Error, - List? Nodes, Guid? ClientId, Guid? ClientObjectId, Dictionary Tags +// 'Nodes' removed when porting from Python: only used in search response ) : StatefulEntityBase(State); [JsonConverter(typeof(ContainerConverter))] diff --git a/src/ApiService/ApiService/OneFuzzTypes/Requests.cs b/src/ApiService/ApiService/OneFuzzTypes/Requests.cs index 8f040ae4c5..3917cbb330 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Requests.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Requests.cs @@ -1,4 +1,5 @@ -using System.Text.Json.Serialization; +using System.ComponentModel.DataAnnotations; +using System.Text.Json.Serialization; namespace Microsoft.OneFuzz.Service; @@ -164,6 +165,37 @@ public record ProxyReset( string Region ); +public record ScalesetCreate( + PoolName PoolName, + string VmSku, + string Image, + string? Region, + [Range(1, long.MaxValue)] + [property: Range(1, long.MaxValue)] + long? Size, + bool SpotInstances, + Dictionary Tags, + bool EphemeralOsDisks = false +); + +public record ScalesetSearch( + Guid? ScalesetId = null, + List? State = null, + bool IncludeAuth = false +); + +public record ScalesetStop( + Guid ScalesetId, + bool Now +); + +public record ScalesetUpdate( + Guid ScalesetId, + [Range(1, long.MaxValue)] + [property: Range(1, long.MaxValue)] + long? Size +); + public record TaskGet(Guid TaskId); public record TaskSearch( diff --git a/src/ApiService/ApiService/OneFuzzTypes/Responses.cs b/src/ApiService/ApiService/OneFuzzTypes/Responses.cs index 7285a827b1..81c92a153b 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Responses.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Responses.cs @@ -105,6 +105,44 @@ public record PoolGetResult( List? ScalesetSummary ) : BaseResponse(); +public record ScalesetSearchResponse( + PoolName PoolName, + Guid ScalesetId, + ScalesetState State, + Authentication? Auth, + string VmSku, + string Image, + string Region, + long Size, + bool? SpotInstances, + bool EmphemeralOsDisks, + bool NeedsConfigUpdate, + Error? Error, + Guid? ClientId, + Guid? ClientObjectId, + Dictionary Tags, + List? Nodes +) : BaseResponse() { + public static ScalesetSearchResponse ForScaleset(Scaleset s) + => new( + PoolName: s.PoolName, + ScalesetId: s.ScalesetId, + State: s.State, + Auth: s.Auth, + VmSku: s.VmSku, + Image: s.Image, + Region: s.Region, + Size: s.Size, + SpotInstances: s.SpotInstances, + EmphemeralOsDisks: s.EphemeralOsDisks, + NeedsConfigUpdate: s.NeedsConfigUpdate, + Error: s.Error, + ClientId: s.ClientId, + ClientObjectId: s.ClientObjectId, + Tags: s.Tags, + Nodes: null); +} + public class BaseResponseConverter : JsonConverter { public override BaseResponse? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) { return null; @@ -130,4 +168,3 @@ VmState State public record ProxyList( List Proxies ); - diff --git a/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs b/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs index 4e3272f69c..4536b5051f 100644 --- a/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs +++ b/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs @@ -19,15 +19,18 @@ public interface IScalesetOperations : IStatefulOrm { Async.Task SetSize(Scaleset scaleset, int size); Async.Task SyncScalesetSize(Scaleset scaleset); - Async.Task SetShutdown(Scaleset scaleset, bool now); Async.Task SetState(Scaleset scaleset, ScalesetState state); + public Async.Task> GetNodes(Scaleset scaleset); + IAsyncEnumerable SearchStates(IEnumerable states); + Async.Task SetShutdown(Scaleset scaleset, bool now); + Async.Task> SetSize(Scaleset scaleset, long size); } public class ScalesetOperations : StatefulOrm, IScalesetOperations { const string SCALESET_LOG_PREFIX = "scalesets: "; - ILogTracer _log; + private readonly ILogTracer _log; public ScalesetOperations(ILogTracer log, IOnefuzzContext context) : base(log, context) { @@ -108,30 +111,23 @@ public async Async.Task SetSize(Scaleset scaleset, int size) { } - static int ScalesetMaxSize(string image) { - // https://docs.microsoft.com/en-us/azure/virtual-machine-scale-sets/ - // virtual-machine-scale-sets-placement-groups#checklist-for-using-large-scale-sets - if (image.StartsWith('/')) - return 600; - else - return 1000; - } - - static int MaxSize(Scaleset scaleset) { - return ScalesetMaxSize(scaleset.Image); - } - - public async Async.Task SetState(Scaleset scaleset, ScalesetState state) { - if (scaleset.State == state) - return scaleset; + async Async.Task> SetState(Scaleset scaleSet, ScalesetState state) { + if (scaleSet.State == state) { + return OneFuzzResult.Ok(scaleSet); + } - if (scaleset.State == ScalesetState.Halt) - return scaleset; + if (scaleSet.State == ScalesetState.Halt) { + // terminal state, unable to change + // TODO: should this throw an exception instead? + return OneFuzzResult.Ok(scaleSet); + } - var updatedScaleSet = scaleset with { State = state }; + var updatedScaleSet = scaleSet with { State = state }; var r = await Update(updatedScaleSet); if (!r.IsOk) { - _log.Error($"Failed to update scaleset {scaleset.ScalesetId} when updating state from {scaleset.State} to {state}"); + var msg = "Failed to update scaleset {scaleSet.ScalesetId} when updating state from {scaleSet.State} to {state}"; + _log.Error(msg); + return OneFuzzResult.Error(ErrorCode.UNABLE_TO_UPDATE, msg); } if (state == ScalesetState.Resize) { @@ -144,14 +140,19 @@ await _context.Events.SendEvent( ); } - return updatedScaleSet; + return OneFuzzResult.Ok(scaleSet); } - async Async.Task SetFailed(Scaleset scaleset, Error error) { - if (scaleset.Error is not null) - return scaleset; + async Async.Task> SetFailed(Scaleset scaleset, Error error) { + if (scaleset.Error is not null) { + return OneFuzzResult.Ok(scaleset); + } var updatedScaleset = await SetState(scaleset with { Error = error }, ScalesetState.CreationFailed); + if (!updatedScaleset.IsOk) { + return updatedScaleset; + } + await _context.Events.SendEvent(new EventScalesetFailed(scaleset.ScalesetId, scaleset.PoolName, error)); return updatedScaleset; } @@ -174,9 +175,9 @@ public async Async.Task UpdateConfigs(Scaleset scaleSet) { var pool = await _context.PoolOperations.GetByName(scaleSet.PoolName); - if (!pool.IsOk || pool.OkV is null) { + if (!pool.IsOk) { _log.Error($"{SCALESET_LOG_PREFIX} unable to find pool during config update. pool:{scaleSet.PoolName}, scaleset_id:{scaleSet.ScalesetId}"); - await SetFailed(scaleSet, pool.ErrorV!); + await SetFailed(scaleSet, pool.ErrorV); return; } @@ -189,15 +190,10 @@ public async Async.Task UpdateConfigs(Scaleset scaleSet) { } } - public async Async.Task SetShutdown(Scaleset scaleset, bool now) { - if (now) { - return await SetState(scaleset, ScalesetState.Halt); - } else { - return await SetState(scaleset, ScalesetState.Shutdown); - } - } + public Async.Task> SetShutdown(Scaleset scaleset, bool now) + => SetState(scaleset, now ? ScalesetState.Halt : ScalesetState.Shutdown); - public async Async.Task Setup(Scaleset scaleset) { + public async Async.Task> Setup(Scaleset scaleset) { //# TODO: How do we pass in SSH configs for Windows? Previously //# This was done as part of the generated per-task setup script. _logTracer.Info($"{SCALESET_LOG_PREFIX} setup. scalset_id: {scaleset.ScalesetId}"); @@ -210,12 +206,14 @@ public async Async.Task Setup(Scaleset scaleset) { if (!result.IsOk) { return await SetFailed(scaleset, result.ErrorV); } + //TODO : why are we saving scaleset here ? var r = await Update(scaleset); if (!r.IsOk) { _logTracer.Error($"Failed to save scaleset {scaleset.ScalesetId} due to {r.ErrorV}"); } - return scaleset; + + return OneFuzzResult.Ok(scaleset); } if (scaleset.Auth is null) { @@ -277,7 +275,12 @@ public async Async.Task Setup(Scaleset scaleset) { _logTracer.Error($"Failed to set identity for scaleset {scaleset.ScalesetId} due to: {result.ErrorV}"); return await SetFailed(scaleset, result.ErrorV); } else { - scaleset = await SetState(scaleset, ScalesetState.Running); + var updateResult = await SetState(scaleset, ScalesetState.Running); + if (!updateResult.IsOk) { + return updateResult.ErrorV; + } + + scaleset = updateResult.OkV; } } @@ -285,7 +288,8 @@ public async Async.Task Setup(Scaleset scaleset) { if (!rr.IsOk) { _logTracer.Error($"Failed to save scale data for scale set: {scaleset.ScalesetId}"); } - return scaleset; + + return OneFuzzResult.Ok(scaleset); } @@ -331,30 +335,30 @@ async Async.Task TryEnableAutoScaling(Scaleset scaleset) { var autoScaleConfig = await _context.AutoScaleOperations.GetSettingsForScaleset(scaleset.ScalesetId); - Azure.Management.Monitor.Models.AutoscaleProfile autoScaleProfile; if (poolQueueUri is null) { var failedToFindQueueUri = OneFuzzResultVoid.Error(ErrorCode.UNABLE_TO_FIND, $"Failed to get pool queue uri for scaleset {scaleset.ScalesetId}"); _logTracer.Error(failedToFindQueueUri.ErrorV.ToString()); return failedToFindQueueUri; - } else { + } - if (autoScaleConfig is null) { - autoScaleProfile = _context.AutoScaleOperations.DeafaultAutoScaleProfile(poolQueueUri!, capacity.Value); - } else { - _logTracer.Info("Using existing auto scale settings from database"); - autoScaleProfile = _context.AutoScaleOperations.CreateAutoScaleProfile( - poolQueueUri!, - autoScaleConfig.Min, - autoScaleConfig.Max, - autoScaleConfig.Default, - autoScaleConfig.ScaleOutAmount, - autoScaleConfig.ScaleOutCoolDown, - autoScaleConfig.ScaleInAmount, - autoScaleConfig.ScaleInCoolDown - ); + AutoscaleProfile autoScaleProfile; + if (autoScaleConfig is null) { + autoScaleProfile = _context.AutoScaleOperations.DefaultAutoScaleProfile(poolQueueUri!, capacity.Value); + } else { + _logTracer.Info("Using existing auto scale settings from database"); + autoScaleProfile = _context.AutoScaleOperations.CreateAutoScaleProfile( + poolQueueUri!, + autoScaleConfig.Min, + autoScaleConfig.Max, + autoScaleConfig.Default, + autoScaleConfig.ScaleOutAmount, + autoScaleConfig.ScaleOutCoolDown, + autoScaleConfig.ScaleInAmount, + autoScaleConfig.ScaleInCoolDown + ); - } } + _logTracer.Info($"Added auto scale resource to scaleset: {scaleset.ScalesetId}"); return await _context.AutoScaleOperations.AddAutoScaleToVmss(scaleset.ScalesetId, autoScaleProfile); } @@ -667,4 +671,48 @@ private async Async.Task ResizeShrink(Scaleset scaleset, long? toRemove) { } } } + + public async Task> GetNodes(Scaleset scaleset) { + // Be in at-least 'setup' before checking for the list of VMs + if (scaleset.State == ScalesetState.Init) { + return new List(); + } + + var (nodes, azureNodes) = await ( + _context.NodeOperations.SearchStates(scaleset.ScalesetId).ToListAsync().AsTask(), + _context.VmssOperations.ListInstanceIds(scaleset.ScalesetId)); + + var result = new List(); + foreach (var (machineId, instanceId) in azureNodes) { + var node = nodes.FirstOrDefault(n => n.MachineId == machineId); + result.Add(new ScalesetNodeState( + MachineId: machineId, + InstanceId: instanceId, + node?.State)); + } + + return result; + } + + public IAsyncEnumerable SearchStates(IEnumerable states) + => QueryAsync(Query.EqualAnyEnum("state", states)); + + public Async.Task> SetSize(Scaleset scaleset, long size) { + var permittedSize = Math.Min(size, MaxSize(scaleset)); + if (permittedSize == scaleset.Size) { + return Async.Task.FromResult(OneFuzzResult.Ok(scaleset)); // nothing to do + } + + scaleset = scaleset with { Size = permittedSize }; + return SetState(scaleset, ScalesetState.Resize); + } + + private static long MaxSize(Scaleset scaleset) { + // https://docs.microsoft.com/en-us/azure/virtual-machine-scale-sets/virtual-machine-scale-sets-placement-groups#checklist-for-using-large-scale-sets + if (scaleset.Image.StartsWith("/", StringComparison.Ordinal)) { + return 600; + } else { + return 1000; + } + } } diff --git a/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs b/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs index d895d7ab2a..9c54ade19e 100644 --- a/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs +++ b/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs @@ -1,5 +1,6 @@ -using Azure; +using Azure; using Azure.Core; +using System.Net; using Azure.ResourceManager.Compute; using Azure.ResourceManager.Compute.Models; using Azure.ResourceManager.Models; @@ -135,36 +136,14 @@ public async Async.Task UpdateExtensions(Guid name, IList> ListInstanceIds(Guid name) { _log.Verbose($"get instance IDs for scaleset {name}"); - var results = new Dictionary(); - VirtualMachineScaleSetResource res; try { - var r = await GetVmssResource(name).GetAsync(); - res = r.Value; - } catch (Exception ex) when (ex is RequestFailedException) { - _log.Verbose($"vm does not exist {name}"); - return results; - } - - if (res is null) { - _log.Verbose($"vm does not exist {name}"); - return results; - } else { - try { - await foreach (var instance in res!.GetVirtualMachineScaleSetVms().AsAsyncEnumerable()) { - if (instance is not null) { - Guid key; - if (Guid.TryParse(instance.Data.VmId, out key)) { - results[key] = instance.Data.InstanceId; - } else { - _log.Error($"failed to convert vmId {instance.Data.VmId} to Guid"); - } - } - } - } catch (Exception ex) when (ex is RequestFailedException || ex is CloudException) { - _log.Exception(ex, $"vm does not exist {name}"); - } + return await GetVmssResource(name) + .GetVirtualMachineScaleSetVms() + .ToDictionaryAsync(vm => Guid.Parse(vm.Data.VmId), vm => vm.Data.InstanceId); + } catch (RequestFailedException ex) when (ex.Status == (int)HttpStatusCode.NotFound || ex.ErrorCode == "NotFound") { + _log.Exception(ex, $"scaleset does not exist: {name}"); + return new Dictionary(); } - return results; } public async Async.Task> GetInstanceVm(Guid name, Guid vmId) { From 9cb190e0b6938946750d0155278deaa0040dfe4d Mon Sep 17 00:00:00 2001 From: George Pollard Date: Mon, 25 Jul 2022 22:46:51 +0000 Subject: [PATCH 02/14] Cleanups --- src/ApiService/ApiService/Functions/TimerWorkers.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/ApiService/ApiService/Functions/TimerWorkers.cs b/src/ApiService/ApiService/Functions/TimerWorkers.cs index be1444fc2f..a149eb4f52 100644 --- a/src/ApiService/ApiService/Functions/TimerWorkers.cs +++ b/src/ApiService/ApiService/Functions/TimerWorkers.cs @@ -15,7 +15,7 @@ public TimerWorkers(ILogTracer log, IOnefuzzContext context) { _nodeOps = context.NodeOperations; } - public async Async.Task ProcessScalesets(Scaleset scaleset) { + public async Async.Task ProcessScalesets(Service.Scaleset scaleset) { _log.Verbose($"checking scaleset for updates: {scaleset.ScalesetId}"); await _scaleSetOps.UpdateConfigs(scaleset); From e950803a8f0410f870895d3a70d5182d98d1ea4b Mon Sep 17 00:00:00 2001 From: George Pollard Date: Tue, 26 Jul 2022 03:40:41 +0000 Subject: [PATCH 03/14] More implementation --- .../ApiService/Functions/Scaleset.cs | 40 ++++++++++++++++-- .../ApiService/OneFuzzTypes/Enums.cs | 42 ++++++++++++++----- .../ApiService/OneFuzzTypes/Responses.cs | 4 +- .../ApiService/onefuzzlib/NodeOperations.cs | 10 ++--- .../onefuzzlib/ScalesetOperations.cs | 9 ++-- src/ApiService/Tests/OrmModelsTest.cs | 5 +-- 6 files changed, 82 insertions(+), 28 deletions(-) diff --git a/src/ApiService/ApiService/Functions/Scaleset.cs b/src/ApiService/ApiService/Functions/Scaleset.cs index cc9fba97fc..3c8e2754c4 100644 --- a/src/ApiService/ApiService/Functions/Scaleset.cs +++ b/src/ApiService/ApiService/Functions/Scaleset.cs @@ -61,7 +61,41 @@ private async Task Patch(HttpRequestData req) { if (!request.IsOk) { return await _context.RequestHandling.NotOk(req, request.ErrorV, "ScalesetUpdate"); } - throw new NotImplementedException(); + + var answer = await _auth.CheckRequireAdmins(req); + if (!answer.IsOk) { + return await _context.RequestHandling.NotOk(req, answer.ErrorV, "ScalesetUpdate"); + } + + var scalesetResult = await _context.ScalesetOperations.GetById(request.OkV.ScalesetId); + if (!scalesetResult.IsOk) { + return await _context.RequestHandling.NotOk(req, scalesetResult.ErrorV, "ScalesetUpdate"); + } + + var scaleset = scalesetResult.OkV; + if (!scaleset.State.CanUpdate()) { + return await _context.RequestHandling.NotOk( + req, + new Error( + Code: ErrorCode.INVALID_REQUEST, + Errors: new[] { $"scaleset must be in one of the following states to update: {string.Join(", ", ScalesetStateHelper.CanUpdateStates)}" }), + "ScalesetUpdate"); + } + + if (request.OkV.Size is long size) { + var resizedScaleset = await _context.ScalesetOperations.SetSize(scaleset, size); + if (resizedScaleset.IsOk) { + scaleset = resizedScaleset.OkV; + } else { + return await _context.RequestHandling.NotOk( + req, + resizedScaleset.ErrorV, + "ScalesetUpdate"); + } + } + + scaleset = scaleset with { Auth = null }; + return await RequestHandling.Ok(req, ScalesetResponse.ForScaleset(scaleset)); } private async Task Get(HttpRequestData req) { @@ -79,7 +113,7 @@ private async Task Get(HttpRequestData req) { var scaleset = scalesetResult.OkV; - var response = ScalesetSearchResponse.ForScaleset(scaleset); + var response = ScalesetResponse.ForScaleset(scaleset); response = response with { Nodes = await _context.ScalesetOperations.GetNodes(scaleset) }; if (!search.IncludeAuth) { response = response with { Auth = null }; @@ -91,7 +125,7 @@ private async Task Get(HttpRequestData req) { var states = search.State ?? Enumerable.Empty(); var scalesets = await _context.ScalesetOperations.SearchStates(states).ToListAsync(); // don't return auths during list actions, only 'get' - var result = scalesets.Select(ss => ScalesetSearchResponse.ForScaleset(ss with { Auth = null })); + var result = scalesets.Select(ss => ScalesetResponse.ForScaleset(ss with { Auth = null })); return await RequestHandling.Ok(req, result); } } diff --git a/src/ApiService/ApiService/OneFuzzTypes/Enums.cs b/src/ApiService/ApiService/OneFuzzTypes/Enums.cs index 93212f929f..a72d817f48 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Enums.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Enums.cs @@ -1,4 +1,4 @@ -using Microsoft.OneFuzz.Service.OneFuzzLib.Orm; +using Microsoft.OneFuzz.Service.OneFuzzLib.Orm; namespace Microsoft.OneFuzz.Service; @@ -138,29 +138,49 @@ public static class JobStateHelper { public static class ScalesetStateHelper { - private static readonly IReadOnlySet _canUpdate = new HashSet { ScalesetState.Init, ScalesetState.Resize }; - private static readonly IReadOnlySet _needsWork = - new HashSet{ + private static readonly HashSet _canUpdate = + new() { + ScalesetState.Init, + ScalesetState.Resize, + }; + + private static readonly HashSet _needsWork = + new() { ScalesetState.Init, ScalesetState.Setup, ScalesetState.Resize, ScalesetState.Shutdown, - ScalesetState.Halt + ScalesetState.Halt, + }; + + private static readonly HashSet _available = + new() { + ScalesetState.Resize, + ScalesetState.Running, + }; + + private static readonly HashSet _resizing = + new() { + ScalesetState.Halt, + ScalesetState.Init, + ScalesetState.Setup, }; - private static readonly IReadOnlySet _available = new HashSet { ScalesetState.Resize, ScalesetState.Running }; - private static readonly IReadOnlySet _resizing = new HashSet { ScalesetState.Halt, ScalesetState.Init, ScalesetState.Setup }; /// set of states that indicate the scaleset can be updated - public static IReadOnlySet CanUpdate => _canUpdate; + public static bool CanUpdate(this ScalesetState state) => _canUpdate.Contains(state); + public static IReadOnlySet CanUpdateStates => _canUpdate; /// set of states that indicate work is needed during eventing - public static IReadOnlySet NeedsWork => _needsWork; + public static bool NeedsWork(this ScalesetState state) => _needsWork.Contains(state); + public static IReadOnlySet NeedsWorkStates => _needsWork; /// set of states that indicate if it's available for work - public static IReadOnlySet Available => _available; + public static bool IsAvailable(this ScalesetState state) => _available.Contains(state); + public static IReadOnlySet AvailableStates => _available; /// set of states that indicate scaleset is resizing - public static IReadOnlySet Resizing => _resizing; + public static bool IsResizing(this ScalesetState state) => _resizing.Contains(state); + public static IReadOnlySet ResizingStates => _resizing; } diff --git a/src/ApiService/ApiService/OneFuzzTypes/Responses.cs b/src/ApiService/ApiService/OneFuzzTypes/Responses.cs index 81c92a153b..f5cbfa9871 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Responses.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Responses.cs @@ -105,7 +105,7 @@ public record PoolGetResult( List? ScalesetSummary ) : BaseResponse(); -public record ScalesetSearchResponse( +public record ScalesetResponse( PoolName PoolName, Guid ScalesetId, ScalesetState State, @@ -123,7 +123,7 @@ public record ScalesetSearchResponse( Dictionary Tags, List? Nodes ) : BaseResponse() { - public static ScalesetSearchResponse ForScaleset(Scaleset s) + public static ScalesetResponse ForScaleset(Scaleset s) => new( PoolName: s.PoolName, ScalesetId: s.ScalesetId, diff --git a/src/ApiService/ApiService/onefuzzlib/NodeOperations.cs b/src/ApiService/ApiService/onefuzzlib/NodeOperations.cs index 9306424eb5..a8ccbe4ad8 100644 --- a/src/ApiService/ApiService/onefuzzlib/NodeOperations.cs +++ b/src/ApiService/ApiService/onefuzzlib/NodeOperations.cs @@ -141,25 +141,25 @@ public async Task CanProcessNewWork(Node node) { if (node.ScalesetId != null) { var scalesetResult = await _context.ScalesetOperations.GetById(node.ScalesetId.Value); - if (!scalesetResult.IsOk || scalesetResult.OkV == null) { + if (!scalesetResult.IsOk) { _logTracer.Info($"can_process_new_work invalid scaleset. scaleset_id:{node.ScalesetId} machine_id:{node.MachineId}"); return false; } - var scaleset = scalesetResult.OkV!; - if (!ScalesetStateHelper.Available.Contains(scaleset.State)) { + var scaleset = scalesetResult.OkV; + if (!scaleset.State.IsAvailable()) { _logTracer.Info($"can_process_new_work scaleset not available for work. scaleset_id:{node.ScalesetId} machine_id:{node.MachineId}"); return false; } } var poolResult = await _context.PoolOperations.GetByName(node.PoolName); - if (!poolResult.IsOk || poolResult.OkV == null) { + if (!poolResult.IsOk) { _logTracer.Info($"can_schedule - invalid pool. pool_name:{node.PoolName} machine_id:{node.MachineId}"); return false; } - var pool = poolResult.OkV!; + var pool = poolResult.OkV; if (!PoolStateHelper.Available.Contains(pool.State)) { _logTracer.Info($"can_schedule - pool is not available for work. pool_name:{node.PoolName} machine_id:{node.MachineId}"); return false; diff --git a/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs b/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs index 4536b5051f..562050eb87 100644 --- a/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs +++ b/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs @@ -1,6 +1,7 @@ using System.Threading.Tasks; using ApiService.OneFuzzLib.Orm; using Azure.ResourceManager.Compute; +using Microsoft.Azure.Management.Monitor.Models; namespace Microsoft.OneFuzz.Service; @@ -108,7 +109,6 @@ public async Async.Task SetSize(Scaleset scaleset, int size) { } else { await ResizeShrink(scaleset, vmssSize - scaleset.Size); } - } async Async.Task> SetState(Scaleset scaleSet, ScalesetState state) { @@ -343,7 +343,7 @@ async Async.Task TryEnableAutoScaling(Scaleset scaleset) { AutoscaleProfile autoScaleProfile; if (autoScaleConfig is null) { - autoScaleProfile = _context.AutoScaleOperations.DefaultAutoScaleProfile(poolQueueUri!, capacity.Value); + autoScaleProfile = _context.AutoScaleOperations.DeafaultAutoScaleProfile(poolQueueUri!, capacity.Value); } else { _logTracer.Info("Using existing auto scale settings from database"); autoScaleProfile = _context.AutoScaleOperations.CreateAutoScaleProfile( @@ -364,7 +364,7 @@ async Async.Task TryEnableAutoScaling(Scaleset scaleset) { } - public async Async.Task Init(Scaleset scaleset) { + public async Async.Task> Init(Scaleset scaleset) { _logTracer.Info($"{SCALESET_LOG_PREFIX} init. scaleset_id:{scaleset.ScalesetId}"); var shrinkQueue = new ShrinkQueue(scaleset.ScalesetId, _context.Queue, _logTracer); await shrinkQueue.Create(); @@ -395,7 +395,8 @@ public async Async.Task Init(Scaleset scaleset) { } else { return await SetState(scaleset, ScalesetState.Setup); } - return scaleset; + + return OneFuzzResult.Ok(scaleset); } public async Async.Task Halt(Scaleset scaleset) { diff --git a/src/ApiService/Tests/OrmModelsTest.cs b/src/ApiService/Tests/OrmModelsTest.cs index f686b44247..f732f020be 100644 --- a/src/ApiService/Tests/OrmModelsTest.cs +++ b/src/ApiService/Tests/OrmModelsTest.cs @@ -220,7 +220,7 @@ public static Gen Task() { public static Gen Scaleset { get; } = from arg in Arb.Generate, - Tuple, Guid?>, + Tuple, Tuple>>>() from poolName in PoolNameGen select new Scaleset( @@ -237,8 +237,7 @@ from poolName in PoolNameGen EphemeralOsDisks: arg.Item2.Item3, NeedsConfigUpdate: arg.Item2.Item4, Error: arg.Item2.Item5, - Nodes: arg.Item2.Item6, - ClientId: arg.Item2.Item7, + ClientId: arg.Item2.Item6, ClientObjectId: arg.Item3.Item1, Tags: arg.Item3.Item2); From 8b87d0711ccbad8c9e97c83736f778b6ecc1762d Mon Sep 17 00:00:00 2001 From: George Pollard Date: Wed, 10 Aug 2022 23:07:36 +0000 Subject: [PATCH 04/14] Finish POST implementation --- .../ApiService/Functions/Scaleset.cs | 83 ++++++++++++++++++- .../ApiService/OneFuzzTypes/Model.cs | 10 +-- .../ApiService/OneFuzzTypes/Requests.cs | 17 +++- src/ApiService/ApiService/onefuzzlib/Creds.cs | 8 ++ .../ApiService/onefuzzlib/Request.cs | 13 ++- .../ApiService/onefuzzlib/VmssOperations.cs | 69 +++++++++++---- 6 files changed, 170 insertions(+), 30 deletions(-) diff --git a/src/ApiService/ApiService/Functions/Scaleset.cs b/src/ApiService/ApiService/Functions/Scaleset.cs index 3c8e2754c4..0549449a19 100644 --- a/src/ApiService/ApiService/Functions/Scaleset.cs +++ b/src/ApiService/ApiService/Functions/Scaleset.cs @@ -53,7 +53,88 @@ private async Task Post(HttpRequestData req) { return await _context.RequestHandling.NotOk(req, request.ErrorV, "ScalesetCreate"); } - throw new NotImplementedException(); + var answer = await _auth.CheckRequireAdmins(req); + if (!answer.IsOk) { + return await _context.RequestHandling.NotOk(req, answer.ErrorV, "ScalesetCreate"); + } + + var create = request.OkV; + // verify the pool exists + var poolResult = await _context.PoolOperations.GetByName(create.PoolName); + if (!poolResult.IsOk) { + return await _context.RequestHandling.NotOk(req, answer.ErrorV, "ScalesetCreate"); + } + + var pool = poolResult.OkV; + if (!pool.Managed) { + return await _context.RequestHandling.NotOk( + req, + new Error( + Code: ErrorCode.UNABLE_TO_CREATE, + Errors: new string[] { "scalesets can only be added to managed pools " }), + context: "ScalesetCreate"); + } + + string region; + if (create.Region is null) { + region = await _context.Creds.GetBaseRegion(); + } else { + var validRegions = await _context.Creds.GetRegions(); + if (!validRegions.Contains(create.Region)) { + return await _context.RequestHandling.NotOk( + req, + new Error( + Code: ErrorCode.UNABLE_TO_CREATE, + Errors: new string[] { "invalid region" }), + context: "ScalesetCreate"); + } + + region = create.Region; + } + + var availableSkus = await _context.VmssOperations.ListAvailableSkus(region); + if (!availableSkus.Contains(create.VmSku)) { + return await _context.RequestHandling.NotOk( + req, + new Error( + Code: ErrorCode.UNABLE_TO_CREATE, + Errors: new string[] { $"The specified VM SKU '{create.VmSku}' is not available in the location ${region}" }), + context: "ScalesetCreate"); + } + + var tags = create.Tags; + var configTags = (await _context.ConfigOperations.Fetch()).VmssTags; + if (configTags is not null) { + foreach (var (key, value) in configTags) { + tags[key] = value; + } + } + + var scaleset = new Service.Scaleset( + ScalesetId: Guid.NewGuid(), + State: ScalesetState.Init, + NeedsConfigUpdate: false, + PoolName: create.PoolName, + VmSku: create.VmSku, + Image: create.Image, + Region: region, + Size: create.Size, + SpotInstances: create.SpotInstances, + EphemeralOsDisks: create.EphemeralOsDisks, + Tags: tags); + + var inserted = await _context.ScalesetOperations.Insert(scaleset); + if (!inserted.IsOk) { + return await _context.RequestHandling.NotOk( + req, + new Error( + Code: ErrorCode.UNABLE_TO_CREATE, + new string[] { $"unable to insert scaleset: {inserted.ErrorV}" } + ), + context: "ScalesetCreate"); + } + + return await RequestHandling.Ok(req, ScalesetResponse.ForScaleset(scaleset)); } private async Task Patch(HttpRequestData req) { diff --git a/src/ApiService/ApiService/OneFuzzTypes/Model.cs b/src/ApiService/ApiService/OneFuzzTypes/Model.cs index 9c0eb34db0..1e63832bd7 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Model.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Model.cs @@ -394,7 +394,6 @@ public record Scaleset( [PartitionKey] PoolName PoolName, [RowKey] Guid ScalesetId, ScalesetState State, - Authentication? Auth, string VmSku, string Image, Region Region, @@ -402,10 +401,11 @@ public record Scaleset( bool? SpotInstances, bool EphemeralOsDisks, bool NeedsConfigUpdate, - Error? Error, - Guid? ClientId, - Guid? ClientObjectId, - Dictionary Tags + Dictionary Tags, + Authentication? Auth = null, + Error? Error = null, + Guid? ClientId = null, + Guid? ClientObjectId = null // 'Nodes' removed when porting from Python: only used in search response ) : StatefulEntityBase(State); diff --git a/src/ApiService/ApiService/OneFuzzTypes/Requests.cs b/src/ApiService/ApiService/OneFuzzTypes/Requests.cs index 3917cbb330..4b939f2108 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Requests.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Requests.cs @@ -170,12 +170,22 @@ public record ScalesetCreate( string VmSku, string Image, string? Region, - [Range(1, long.MaxValue)] [property: Range(1, long.MaxValue)] - long? Size, + long Size, bool SpotInstances, Dictionary Tags, - bool EphemeralOsDisks = false + bool EphemeralOsDisks = false, + AutoScaleOptions? AutoScale = null +); + +public record AutoScaleOptions( + [property: Range(0, long.MaxValue)] long Min, + [property: Range(1, long.MaxValue)] long Max, + [property: Range(0, long.MaxValue)] long Default, + [property: Range(1, long.MaxValue)] long ScaleOutAmount, + [property: Range(1, long.MaxValue)] long ScaleOutCooldown, + [property: Range(1, long.MaxValue)] long ScaleInAmount, + [property: Range(1, long.MaxValue)] long ScaleInCooldown ); public record ScalesetSearch( @@ -191,7 +201,6 @@ bool Now public record ScalesetUpdate( Guid ScalesetId, - [Range(1, long.MaxValue)] [property: Range(1, long.MaxValue)] long? Size ); diff --git a/src/ApiService/ApiService/onefuzzlib/Creds.cs b/src/ApiService/ApiService/onefuzzlib/Creds.cs index d515178c00..8b16d430e6 100644 --- a/src/ApiService/ApiService/onefuzzlib/Creds.cs +++ b/src/ApiService/ApiService/onefuzzlib/Creds.cs @@ -24,6 +24,8 @@ public interface ICreds { public ResourceGroupResource GetResourceGroupResource(); + public SubscriptionResource GetSubscriptionResource(); + public Async.Task GetBaseRegion(); public Uri GetInstanceUrl(); @@ -91,6 +93,11 @@ public ResourceGroupResource GetResourceGroupResource() { return ArmClient.GetResourceGroupResource(resourceId); } + public SubscriptionResource GetSubscriptionResource() { + var id = SubscriptionResource.CreateResourceIdentifier(GetSubscription()); + return ArmClient.GetSubscriptionResource(id); + } + public Async.Task GetBaseRegion() { return _cache.GetOrCreateAsync(nameof(GetBaseRegion), async _ => { var rg = await ArmClient.GetResourceGroupResource(GetResourceGroupResourceIdentifier()).GetAsync(); @@ -188,6 +195,7 @@ public Task> GetRegions() .Select(x => x.Name) .ToListAsync(); }); + } diff --git a/src/ApiService/ApiService/onefuzzlib/Request.cs b/src/ApiService/ApiService/onefuzzlib/Request.cs index bcb01940eb..23eca9e47a 100644 --- a/src/ApiService/ApiService/onefuzzlib/Request.cs +++ b/src/ApiService/ApiService/onefuzzlib/Request.cs @@ -1,4 +1,5 @@ -using System.Net; +using System.ComponentModel.DataAnnotations; +using System.Net; using System.Text.Json; using System.Text.Json.Nodes; using Faithlife.Utility; @@ -35,7 +36,15 @@ public static async Async.Task> ParseRequest(HttpRequestData try { var t = await req.ReadFromJsonAsync(); if (t != null) { - return OneFuzzResult.Ok(t); + var validationContext = new ValidationContext(t); + var validationResults = new List(); + if (Validator.TryValidateObject(t, validationContext, validationResults, true)) { + return OneFuzzResult.Ok(t); + } else { + return new Error( + Code: ErrorCode.INVALID_REQUEST, + Errors: validationResults.Select(vr => vr.ToString()).ToArray()); + } } } catch (Exception e) { exception = e; diff --git a/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs b/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs index 9c54ade19e..ced339aeff 100644 --- a/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs +++ b/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs @@ -4,7 +4,9 @@ using Azure.ResourceManager.Compute; using Azure.ResourceManager.Compute.Models; using Azure.ResourceManager.Models; +using Microsoft.Extensions.Caching.Memory; using Microsoft.Rest.Azure; +using Azure.Data.Tables; namespace Microsoft.OneFuzz.Service; @@ -14,6 +16,8 @@ public interface IVmssOperations { Async.Task UpdateExtensions(Guid name, IList extensions); Async.Task GetVmss(Guid name); + Async.Task> ListAvailableSkus(string region); + Async.Task DeleteVmss(Guid name, bool? forceDeletion = null); Async.Task> ListInstanceIds(Guid name); @@ -38,17 +42,18 @@ Async.Task CreateVmss( } public class VmssOperations : IVmssOperations { + private readonly ILogTracer _log; + private readonly ICreds _creds; + private readonly IImageOperations _imageOps; + private readonly IServiceConfig _serviceConfig; + private readonly IMemoryCache _cache; - readonly ILogTracer _log; - readonly ICreds _creds; - readonly IImageOperations _imageOps; - readonly IServiceConfig _serviceConfig; - - public VmssOperations(ILogTracer log, IOnefuzzContext context) { + public VmssOperations(ILogTracer log, IOnefuzzContext context, IMemoryCache cache) { _log = log; _creds = context.Creds; _imageOps = context.ImageOperations; _serviceConfig = context.ServiceConfiguration; + _cache = cache; } public async Async.Task DeleteVmss(Guid name, bool? forceDeletion = null) { @@ -202,18 +207,18 @@ public async Async.Task UpdateScaleInProtection(Guid name, Gu } public async Async.Task CreateVmss( - string location, - Guid name, - string vmSku, - long vmCount, - string image, - string networkId, - bool? spotInstance, - bool ephemeralOsDisks, - IList? extensions, - string password, - string sshPublicKey, - IDictionary tags) { + string location, + Guid name, + string vmSku, + long vmCount, + string image, + string networkId, + bool? spotInstance, + bool ephemeralOsDisks, + IList? extensions, + string password, + string sshPublicKey, + IDictionary tags) { var vmss = await GetVmss(name); if (vmss is not null) { return OneFuzzResultVoid.Ok; @@ -319,4 +324,32 @@ public async Async.Task CreateVmss( return OneFuzzResultVoid.Error(ErrorCode.VM_CREATE_FAILED, new[] { ex.Message }); } } + + public Async.Task> ListAvailableSkus(string region) + => _cache.GetOrCreateAsync>($"compute-skus-{region}", async entry => { + entry.SetAbsoluteExpiration(TimeSpan.FromMinutes(10)); + + var sub = _creds.GetSubscriptionResource(); + var skus = sub.GetResourceSkusAsync(filter: TableClient.CreateQueryFilter($"location eq '{region}'")); + + var skuNames = new List(); + await foreach (var sku in skus) { + var available = true; + if (sku.Restrictions is not null) { + foreach (var restriction in sku.Restrictions) { + if (restriction.RestrictionsType == ResourceSkuRestrictionsType.Location && + restriction.Values.Contains(region, StringComparer.OrdinalIgnoreCase)) { + available = false; + break; + } + } + } + + if (available) { + skuNames.Add(sku.Name); + } + } + + return skuNames; + }); } From 696e3f83458a84b15c54e52d934c6817857e4d84 Mon Sep 17 00:00:00 2001 From: George Pollard Date: Wed, 10 Aug 2022 23:37:41 +0000 Subject: [PATCH 05/14] Fix test build failure --- src/ApiService/IntegrationTests/Fakes/TestCreds.cs | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/ApiService/IntegrationTests/Fakes/TestCreds.cs b/src/ApiService/IntegrationTests/Fakes/TestCreds.cs index e275b34ef7..14b09a67ff 100644 --- a/src/ApiService/IntegrationTests/Fakes/TestCreds.cs +++ b/src/ApiService/IntegrationTests/Fakes/TestCreds.cs @@ -47,6 +47,10 @@ public ResourceGroupResource GetResourceGroupResource() { throw new NotImplementedException(); } + public SubscriptionResource GetSubscriptionResource() { + throw new NotImplementedException(); + } + public ResourceIdentifier GetResourceGroupResourceIdentifier() { throw new NotImplementedException(); } From faa00fdcf408cc6f58696c2c0b17d2e921eab750 Mon Sep 17 00:00:00 2001 From: George Pollard Date: Wed, 10 Aug 2022 23:44:23 +0000 Subject: [PATCH 06/14] Basic test setup --- src/ApiService/IntegrationTests/PoolTests.cs | 4 +- .../IntegrationTests/ScalesetTests.cs | 68 +++++++++++++++++++ 2 files changed, 69 insertions(+), 3 deletions(-) create mode 100644 src/ApiService/IntegrationTests/ScalesetTests.cs diff --git a/src/ApiService/IntegrationTests/PoolTests.cs b/src/ApiService/IntegrationTests/PoolTests.cs index 647a0632db..c3405d66f3 100644 --- a/src/ApiService/IntegrationTests/PoolTests.cs +++ b/src/ApiService/IntegrationTests/PoolTests.cs @@ -1,6 +1,4 @@ - - -using System; +using System; using System.Collections.Generic; using System.Linq; using System.Net; diff --git a/src/ApiService/IntegrationTests/ScalesetTests.cs b/src/ApiService/IntegrationTests/ScalesetTests.cs new file mode 100644 index 0000000000..c339580ba6 --- /dev/null +++ b/src/ApiService/IntegrationTests/ScalesetTests.cs @@ -0,0 +1,68 @@ +using System; +using System.Linq; +using System.Net; +using IntegrationTests.Fakes; +using Microsoft.OneFuzz.Service; +using Xunit; +using Xunit.Abstractions; +using Async = System.Threading.Tasks; +using ScalesetFunction = Microsoft.OneFuzz.Service.Functions.Scaleset; + +namespace IntegrationTests.Functions; + +[Trait("Category", "Live")] +public class AzureStorageScalesetTest : ScalesetTestBase { + public AzureStorageScalesetTest(ITestOutputHelper output) + : base(output, Integration.AzureStorage.FromEnvironment()) { } +} + +public class AzuriteScalesetTest : ScalesetTestBase { + public AzuriteScalesetTest(ITestOutputHelper output) + : base(output, new Integration.AzuriteStorage()) { } +} + +public abstract class ScalesetTestBase : FunctionTestBase { + public ScalesetTestBase(ITestOutputHelper output, IStorage storage) + : base(output, storage) { } + + [Theory] + [InlineData("POST", RequestType.Agent)] + [InlineData("POST", RequestType.NoAuthorization)] + [InlineData("PATCH", RequestType.Agent)] + [InlineData("PATCH", RequestType.NoAuthorization)] + [InlineData("GET", RequestType.Agent)] + [InlineData("GET", RequestType.NoAuthorization)] + [InlineData("DELETE", RequestType.Agent)] + [InlineData("DELETE", RequestType.NoAuthorization)] + public async Async.Task UserAuthorization_IsRequired(string method, RequestType authType) { + var auth = new TestEndpointAuthorization(authType, Logger, Context); + var func = new ScalesetFunction(Logger, auth, Context); + var result = await func.Run(TestHttpRequestData.Empty(method)); + Assert.Equal(HttpStatusCode.Unauthorized, result.StatusCode); + } + + [Fact] + public async Async.Task Search_SpecificScaleset_ReturnsErrorIfNoneFound() { + var auth = new TestEndpointAuthorization(RequestType.User, Logger, Context); + + var req = new ScalesetSearch(ScalesetId: Guid.NewGuid()); + var func = new ScalesetFunction(Logger, auth, Context); + var result = await func.Run(TestHttpRequestData.FromJson("GET", req)); + + Assert.Equal(HttpStatusCode.BadRequest, result.StatusCode); + var err = BodyAs(result); + Assert.Equal("unable to find scaleset", err.Errors?.Single()); + } + + [Fact] + public async Async.Task Search_AllScalesets_ReturnsEmptyIfNoneFound() { + var auth = new TestEndpointAuthorization(RequestType.User, Logger, Context); + + var req = new ScalesetSearch(); + var func = new ScalesetFunction(Logger, auth, Context); + var result = await func.Run(TestHttpRequestData.FromJson("GET", req)); + + Assert.Equal(HttpStatusCode.OK, result.StatusCode); + Assert.Equal("[]", BodyAsString(result)); + } +} From 53e8e06f1a64ad140e2a04553dc9146f65cd3f15 Mon Sep 17 00:00:00 2001 From: George Pollard Date: Wed, 10 Aug 2022 23:47:35 +0000 Subject: [PATCH 07/14] format --- src/ApiService/IntegrationTests/ScalesetTests.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/ApiService/IntegrationTests/ScalesetTests.cs b/src/ApiService/IntegrationTests/ScalesetTests.cs index c339580ba6..ed27ce60b2 100644 --- a/src/ApiService/IntegrationTests/ScalesetTests.cs +++ b/src/ApiService/IntegrationTests/ScalesetTests.cs @@ -1,4 +1,4 @@ -using System; +using System; using System.Linq; using System.Net; using IntegrationTests.Fakes; From f82029e43e89f32c35cfb3b7f555fd30c6721072 Mon Sep 17 00:00:00 2001 From: George Pollard Date: Mon, 15 Aug 2022 01:37:27 +0000 Subject: [PATCH 08/14] Add key generation --- src/ApiService/ApiService/Functions/Scaleset.cs | 16 +++++++++++++++- 1 file changed, 15 insertions(+), 1 deletion(-) diff --git a/src/ApiService/ApiService/Functions/Scaleset.cs b/src/ApiService/ApiService/Functions/Scaleset.cs index 0549449a19..36bad6588a 100644 --- a/src/ApiService/ApiService/Functions/Scaleset.cs +++ b/src/ApiService/ApiService/Functions/Scaleset.cs @@ -1,4 +1,5 @@ -using System.Threading.Tasks; +using System.Security.Cryptography; +using System.Threading.Tasks; using Microsoft.Azure.Functions.Worker; using Microsoft.Azure.Functions.Worker.Http; @@ -114,6 +115,7 @@ private async Task Post(HttpRequestData req) { ScalesetId: Guid.NewGuid(), State: ScalesetState.Init, NeedsConfigUpdate: false, + Auth: GenerateAuthentication(), PoolName: create.PoolName, VmSku: create.VmSku, Image: create.Image, @@ -137,6 +139,18 @@ private async Task Post(HttpRequestData req) { return await RequestHandling.Ok(req, ScalesetResponse.ForScaleset(scaleset)); } + private static Authentication GenerateAuthentication() { + using var rsa = RSA.Create(2048); + var privateKey = rsa.ExportRSAPrivateKey(); + var publicKey = rsa.ExportRSAPublicKey(); + var formattedPrivateKey = $"-----BEGIN RSA PRIVATE KEY-----\n{Convert.ToBase64String(privateKey)}\n-----END RSA PRIVATE KEY-----\n"; + var formattedPublicKey = $"ssh-rsa {Convert.ToBase64String(publicKey)} onefuzz-generated-key"; + return new Authentication( + Password: Guid.NewGuid().ToString(), + PublicKey: formattedPublicKey, + PrivateKey: formattedPrivateKey); + } + private async Task Patch(HttpRequestData req) { var request = await RequestHandling.ParseRequest(req); if (!request.IsOk) { From 191cc79e6170ec361625c574fd596c35b20d99d7 Mon Sep 17 00:00:00 2001 From: George Pollard Date: Mon, 15 Aug 2022 03:35:19 +0000 Subject: [PATCH 09/14] Generate Public Key format --- .../ApiService/Functions/Scaleset.cs | 35 +++++++++++++++++-- 1 file changed, 33 insertions(+), 2 deletions(-) diff --git a/src/ApiService/ApiService/Functions/Scaleset.cs b/src/ApiService/ApiService/Functions/Scaleset.cs index 36bad6588a..8ad8b6f403 100644 --- a/src/ApiService/ApiService/Functions/Scaleset.cs +++ b/src/ApiService/ApiService/Functions/Scaleset.cs @@ -1,4 +1,9 @@ -using System.Security.Cryptography; +using System; +using System.Buffers.Binary; +using System.Diagnostics; +using System.IO; +using System.Security.Cryptography; +using System.Text; using System.Threading.Tasks; using Microsoft.Azure.Functions.Worker; using Microsoft.Azure.Functions.Worker.Http; @@ -142,15 +147,41 @@ private async Task Post(HttpRequestData req) { private static Authentication GenerateAuthentication() { using var rsa = RSA.Create(2048); var privateKey = rsa.ExportRSAPrivateKey(); - var publicKey = rsa.ExportRSAPublicKey(); var formattedPrivateKey = $"-----BEGIN RSA PRIVATE KEY-----\n{Convert.ToBase64String(privateKey)}\n-----END RSA PRIVATE KEY-----\n"; + + var publicKey = BuildPublicKey(rsa); var formattedPublicKey = $"ssh-rsa {Convert.ToBase64String(publicKey)} onefuzz-generated-key"; + return new Authentication( Password: Guid.NewGuid().ToString(), PublicKey: formattedPublicKey, PrivateKey: formattedPrivateKey); } + private static ReadOnlySpan SSHRSABytes => new byte[] { (byte)'s', (byte)'s', (byte)'h', (byte)'-', (byte)'r', (byte)'s', (byte)'a' }; + + private static byte[] BuildPublicKey(RSA rsa) { + static Span WriteLengthPrefixedBytes(ReadOnlySpan src, Span dest) { + BinaryPrimitives.WriteInt32BigEndian(dest, src.Length); + dest = dest[sizeof(int)..]; + src.CopyTo(dest); + return dest[src.Length..]; + } + + var parameters = rsa.ExportParameters(includePrivateParameters: false); + + // public key format is "ssh-rsa", exponent, modulus, all written + // as (big-endian) length-prefixed bytes + var result = new byte[sizeof(int) + SSHRSABytes.Length + sizeof(int) + parameters.Modulus!.Length + sizeof(int) + parameters.Exponent!.Length]; + var spanResult = result.AsSpan(); + spanResult = WriteLengthPrefixedBytes(SSHRSABytes, spanResult); + spanResult = WriteLengthPrefixedBytes(parameters.Exponent, spanResult); + spanResult = WriteLengthPrefixedBytes(parameters.Modulus, spanResult); + Debug.Assert(spanResult.Length == 0); + + return result; + } + private async Task Patch(HttpRequestData req) { var request = await RequestHandling.ParseRequest(req); if (!request.IsOk) { From 875664012cfdf0fea55efe3f1feb83447c6305ac Mon Sep 17 00:00:00 2001 From: George Pollard Date: Mon, 15 Aug 2022 03:37:18 +0000 Subject: [PATCH 10/14] Format --- src/ApiService/ApiService/Functions/Scaleset.cs | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/src/ApiService/ApiService/Functions/Scaleset.cs b/src/ApiService/ApiService/Functions/Scaleset.cs index 8ad8b6f403..dc939f5ba3 100644 --- a/src/ApiService/ApiService/Functions/Scaleset.cs +++ b/src/ApiService/ApiService/Functions/Scaleset.cs @@ -1,9 +1,6 @@ -using System; -using System.Buffers.Binary; +using System.Buffers.Binary; using System.Diagnostics; -using System.IO; using System.Security.Cryptography; -using System.Text; using System.Threading.Tasks; using Microsoft.Azure.Functions.Worker; using Microsoft.Azure.Functions.Worker.Http; From 6d60e94a25d2166676a6a5c1bad33d6e45084976 Mon Sep 17 00:00:00 2001 From: George Pollard Date: Mon, 15 Aug 2022 23:10:50 +0000 Subject: [PATCH 11/14] Add AutoScale support --- src/ApiService/ApiService/Functions/Scaleset.cs | 14 ++++++++++++++ src/ApiService/ApiService/OneFuzzTypes/Model.cs | 11 +++++++++++ src/ApiService/ApiService/Program.cs | 1 + .../onefuzzlib/AutoScaleOperations.cs | 17 +++++++++++++++++ .../ApiService/onefuzzlib/OnefuzzContext.cs | 4 +--- 5 files changed, 44 insertions(+), 3 deletions(-) create mode 100644 src/ApiService/ApiService/onefuzzlib/AutoScaleOperations.cs diff --git a/src/ApiService/ApiService/Functions/Scaleset.cs b/src/ApiService/ApiService/Functions/Scaleset.cs index dc939f5ba3..fdea15d096 100644 --- a/src/ApiService/ApiService/Functions/Scaleset.cs +++ b/src/ApiService/ApiService/Functions/Scaleset.cs @@ -138,6 +138,20 @@ private async Task Post(HttpRequestData req) { context: "ScalesetCreate"); } + if (create.AutoScale is AutoScaleOptions options) { + var autoScale = new AutoScale( + scaleset.ScalesetId, + Min: options.Min, + Max: options.Max, + Default: options.Default, + ScaleOutAmount: options.ScaleOutAmount, + ScaleOutCooldown: options.ScaleOutCooldown, + ScaleInAmount: options.ScaleInAmount, + ScaleInCooldown: options.ScaleInCooldown); + + await _context.AutoScaleOperations.Insert(autoScale); + } + return await RequestHandling.Ok(req, ScalesetResponse.ForScaleset(scaleset)); } diff --git a/src/ApiService/ApiService/OneFuzzTypes/Model.cs b/src/ApiService/ApiService/OneFuzzTypes/Model.cs index 1e63832bd7..404938d0a3 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Model.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Model.cs @@ -409,6 +409,17 @@ public record Scaleset( // 'Nodes' removed when porting from Python: only used in search response ) : StatefulEntityBase(State); +public record AutoScale( + [PartitionKey] Guid ScalesetId, + long Min, + long Max, + long Default, + long ScaleOutAmount, + long ScaleOutCooldown, + long ScaleInAmount, + long ScaleInCooldown +) : EntityBase; + [JsonConverter(typeof(ContainerConverter))] public record Container(string ContainerName) { public string ContainerName { get; } = ContainerName.All(c => char.IsLetterOrDigit(c) || c == '-') ? ContainerName : throw new ArgumentException("Container name must have only numbers, letters or dashes"); diff --git a/src/ApiService/ApiService/Program.cs b/src/ApiService/ApiService/Program.cs index a001c6636c..f7fd2c2df0 100644 --- a/src/ApiService/ApiService/Program.cs +++ b/src/ApiService/ApiService/Program.cs @@ -100,6 +100,7 @@ public async static Async.Task Main() { .AddScoped() .AddScoped() .AddScoped() + .AddScoped() .AddSingleton() .AddSingleton() diff --git a/src/ApiService/ApiService/onefuzzlib/AutoScaleOperations.cs b/src/ApiService/ApiService/onefuzzlib/AutoScaleOperations.cs new file mode 100644 index 0000000000..137db4b0d2 --- /dev/null +++ b/src/ApiService/ApiService/onefuzzlib/AutoScaleOperations.cs @@ -0,0 +1,17 @@ +using System.Threading.Tasks; +using ApiService.OneFuzzLib.Orm; + +namespace Microsoft.OneFuzz.Service; + +public interface IAutoScaleOperations { + public Task> Insert(AutoScale autoScale); + public Task GetSettingsForScaleset(Guid scalesetId); +} + +public class AutoScaleOperations : Orm, IAutoScaleOperations { + public AutoScaleOperations(ILogTracer logTracer, IOnefuzzContext context) + : base(logTracer, context) { } + + public async Task GetSettingsForScaleset(Guid scalesetId) + => await QueryAsync(Query.PartitionKey(scalesetId.ToString())).SingleAsync(); +} diff --git a/src/ApiService/ApiService/onefuzzlib/OnefuzzContext.cs b/src/ApiService/ApiService/onefuzzlib/OnefuzzContext.cs index c1dda34be6..52f357ac9f 100644 --- a/src/ApiService/ApiService/onefuzzlib/OnefuzzContext.cs +++ b/src/ApiService/ApiService/onefuzzlib/OnefuzzContext.cs @@ -3,7 +3,7 @@ using Microsoft.Extensions.DependencyInjection; public interface IOnefuzzContext { - + IAutoScaleOperations AutoScaleOperations { get; } IConfig Config { get; } IConfigOperations ConfigOperations { get; } IContainers Containers { get; } @@ -40,8 +40,6 @@ public interface IOnefuzzContext { INsgOperations NsgOperations { get; } ISubnet Subnet { get; } IImageOperations ImageOperations { get; } - - IAutoScaleOperations AutoScaleOperations { get; } } public class OnefuzzContext : IOnefuzzContext { From 942e738004b02d8e2da4f95728275719ac180372 Mon Sep 17 00:00:00 2001 From: George Pollard Date: Mon, 15 Aug 2022 23:19:44 +0000 Subject: [PATCH 12/14] Fix tests --- src/ApiService/IntegrationTests/Fakes/TestContext.cs | 1 + 1 file changed, 1 insertion(+) diff --git a/src/ApiService/IntegrationTests/Fakes/TestContext.cs b/src/ApiService/IntegrationTests/Fakes/TestContext.cs index 14455e70f4..d3d6590c3a 100644 --- a/src/ApiService/IntegrationTests/Fakes/TestContext.cs +++ b/src/ApiService/IntegrationTests/Fakes/TestContext.cs @@ -76,6 +76,7 @@ public Async.Task InsertAll(params EntityBase[] objs) public IConfig Config => throw new System.NotImplementedException(); + public IAutoScaleOperations AutoScaleOperations => throw new NotImplementedException(); public IDiskOperations DiskOperations => throw new System.NotImplementedException(); From 11e0d3a66e7eb789f3f69b014239df3482f0840a Mon Sep 17 00:00:00 2001 From: George Pollard Date: Tue, 16 Aug 2022 01:48:02 +0000 Subject: [PATCH 13/14] Fixups after rebase --- .../ApiService/Functions/Scaleset.cs | 10 +-- .../ApiService/OneFuzzTypes/Model.cs | 18 +---- .../ApiService/onefuzzlib/AutoScale.cs | 72 +++++-------------- .../onefuzzlib/AutoScaleOperations.cs | 17 ----- .../onefuzzlib/ScalesetOperations.cs | 59 +++++++-------- .../IntegrationTests/Fakes/TestContext.cs | 2 - 6 files changed, 46 insertions(+), 132 deletions(-) delete mode 100644 src/ApiService/ApiService/onefuzzlib/AutoScaleOperations.cs diff --git a/src/ApiService/ApiService/Functions/Scaleset.cs b/src/ApiService/ApiService/Functions/Scaleset.cs index fdea15d096..e2bb3ad403 100644 --- a/src/ApiService/ApiService/Functions/Scaleset.cs +++ b/src/ApiService/ApiService/Functions/Scaleset.cs @@ -220,15 +220,7 @@ private async Task Patch(HttpRequestData req) { } if (request.OkV.Size is long size) { - var resizedScaleset = await _context.ScalesetOperations.SetSize(scaleset, size); - if (resizedScaleset.IsOk) { - scaleset = resizedScaleset.OkV; - } else { - return await _context.RequestHandling.NotOk( - req, - resizedScaleset.ErrorV, - "ScalesetUpdate"); - } + scaleset = await _context.ScalesetOperations.SetSize(scaleset, size); } scaleset = scaleset with { Auth = null }; diff --git a/src/ApiService/ApiService/OneFuzzTypes/Model.cs b/src/ApiService/ApiService/OneFuzzTypes/Model.cs index 404938d0a3..855d5bf1de 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Model.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Model.cs @@ -378,11 +378,10 @@ public record AutoScale( long Max, long Default, long ScaleOutAmount, - long ScaleOutCoolDown, + long ScaleOutCooldown, long ScaleInAmount, - long ScaleInCoolDown - ) : EntityBase(); - + long ScaleInCooldown +) : EntityBase; public record ScalesetNodeState( Guid MachineId, @@ -409,17 +408,6 @@ public record Scaleset( // 'Nodes' removed when porting from Python: only used in search response ) : StatefulEntityBase(State); -public record AutoScale( - [PartitionKey] Guid ScalesetId, - long Min, - long Max, - long Default, - long ScaleOutAmount, - long ScaleOutCooldown, - long ScaleInAmount, - long ScaleInCooldown -) : EntityBase; - [JsonConverter(typeof(ContainerConverter))] public record Container(string ContainerName) { public string ContainerName { get; } = ContainerName.All(c => char.IsLetterOrDigit(c) || c == '-') ? ContainerName : throw new ArgumentException("Container name must have only numbers, letters or dashes"); diff --git a/src/ApiService/ApiService/onefuzzlib/AutoScale.cs b/src/ApiService/ApiService/onefuzzlib/AutoScale.cs index fcd308f803..f85935d5ef 100644 --- a/src/ApiService/ApiService/onefuzzlib/AutoScale.cs +++ b/src/ApiService/ApiService/onefuzzlib/AutoScale.cs @@ -4,18 +4,10 @@ namespace Microsoft.OneFuzz.Service; public interface IAutoScaleOperations { - Async.Task Create( - Guid scalesetId, - long minAmount, - long maxAmount, - long defaultAmount, - long scaleOutAmount, - long scaleOutCooldown, - long scaleInAmount, - long scaleInCooldown); - Async.Task GetSettingsForScaleset(Guid scalesetId); + public Async.Task> Insert(AutoScale autoScale); + public Async.Task GetSettingsForScaleset(Guid scalesetId); Azure.Management.Monitor.Models.AutoscaleProfile CreateAutoScaleProfile( string queueUri, @@ -27,7 +19,8 @@ Azure.Management.Monitor.Models.AutoscaleProfile CreateAutoScaleProfile( long scaleInAmount, double scaleInCooldownMinutes); - Azure.Management.Monitor.Models.AutoscaleProfile DeafaultAutoScaleProfile(string queueUri, long scaleSetSize); + Azure.Management.Monitor.Models.AutoscaleProfile DefaultAutoScaleProfile(string queueUri, long scaleSetSize); + Async.Task AddAutoScaleToVmss(Guid vmss, Azure.Management.Monitor.Models.AutoscaleProfile autoScaleProfile); } @@ -35,43 +28,10 @@ Azure.Management.Monitor.Models.AutoscaleProfile CreateAutoScaleProfile( public class AutoScaleOperations : Orm, IAutoScaleOperations { public AutoScaleOperations(ILogTracer log, IOnefuzzContext context) - : base(log, context) { - - } - - public async Async.Task Create( - Guid scalesetId, - long minAmount, - long maxAmount, - long defaultAmount, - long scaleOutAmount, - long scaleOutCooldown, - long scaleInAmount, - long scaleInCooldown) { - - var entry = new AutoScale( - scalesetId, - Min: minAmount, - Max: maxAmount, - Default: defaultAmount, - ScaleOutAmount: scaleOutAmount, - ScaleOutCoolDown: scaleOutCooldown, - ScaleInAmount: scaleInAmount, - ScaleInCoolDown: scaleInCooldown - ); - - var r = await Insert(entry); - if (!r.IsOk) { - _logTracer.Error($"Failed to save auto-scale record for scaleset ID: {scalesetId}, minAmount: {minAmount}, maxAmount: {maxAmount}, defaultAmount: {defaultAmount}, scaleOutAmount: {scaleOutAmount}, scaleOutCooldown: {scaleOutCooldown}, scaleInAmount: {scaleInAmount}, scaleInCooldown: {scaleInCooldown}"); - } - return entry; - } - - public async Async.Task GetSettingsForScaleset(Guid scalesetId) { - var autoscale = await GetEntityAsync(scalesetId.ToString(), scalesetId.ToString()); - return autoscale; - } + : base(log, context) { } + public Async.Task GetSettingsForScaleset(Guid scalesetId) + => GetEntityAsync(scalesetId.ToString(), scalesetId.ToString()); public async Async.Task AddAutoScaleToVmss(Guid vmss, Azure.Management.Monitor.Models.AutoscaleProfile autoScaleProfile) { _logTracer.Info($"Checking scaleset {vmss} for existing auto scale resource"); @@ -136,14 +96,14 @@ public async Async.Task AddAutoScaleToVmss(Guid vmss, Azure.M //TODO: Do this using bicep template public Azure.Management.Monitor.Models.AutoscaleProfile CreateAutoScaleProfile( - string queueUri, - long minAmount, - long maxAmount, - long defaultAmount, - long scaleOutAmount, - double scaleOutCooldownMinutes, - long scaleInAmount, - double scaleInCooldownMinutes) { + string queueUri, + long minAmount, + long maxAmount, + long defaultAmount, + long scaleOutAmount, + double scaleOutCooldownMinutes, + long scaleInAmount, + double scaleInCooldownMinutes) { var rules = new[] { //Scale out @@ -200,7 +160,7 @@ public Azure.Management.Monitor.Models.AutoscaleProfile CreateAutoScaleProfile( } - public Azure.Management.Monitor.Models.AutoscaleProfile DeafaultAutoScaleProfile(string queueUri, long scaleSetSize) { + public Azure.Management.Monitor.Models.AutoscaleProfile DefaultAutoScaleProfile(string queueUri, long scaleSetSize) { return CreateAutoScaleProfile(queueUri, 1L, scaleSetSize, scaleSetSize, 1, 10.0, 1, 5.0); } diff --git a/src/ApiService/ApiService/onefuzzlib/AutoScaleOperations.cs b/src/ApiService/ApiService/onefuzzlib/AutoScaleOperations.cs deleted file mode 100644 index 137db4b0d2..0000000000 --- a/src/ApiService/ApiService/onefuzzlib/AutoScaleOperations.cs +++ /dev/null @@ -1,17 +0,0 @@ -using System.Threading.Tasks; -using ApiService.OneFuzzLib.Orm; - -namespace Microsoft.OneFuzz.Service; - -public interface IAutoScaleOperations { - public Task> Insert(AutoScale autoScale); - public Task GetSettingsForScaleset(Guid scalesetId); -} - -public class AutoScaleOperations : Orm, IAutoScaleOperations { - public AutoScaleOperations(ILogTracer logTracer, IOnefuzzContext context) - : base(logTracer, context) { } - - public async Task GetSettingsForScaleset(Guid scalesetId) - => await QueryAsync(Query.PartitionKey(scalesetId.ToString())).SingleAsync(); -} diff --git a/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs b/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs index 562050eb87..6b26d92462 100644 --- a/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs +++ b/src/ApiService/ApiService/onefuzzlib/ScalesetOperations.cs @@ -24,8 +24,8 @@ public interface IScalesetOperations : IStatefulOrm { Async.Task SetState(Scaleset scaleset, ScalesetState state); public Async.Task> GetNodes(Scaleset scaleset); IAsyncEnumerable SearchStates(IEnumerable states); - Async.Task SetShutdown(Scaleset scaleset, bool now); - Async.Task> SetSize(Scaleset scaleset, long size); + Async.Task SetShutdown(Scaleset scaleset, bool now); + Async.Task SetSize(Scaleset scaleset, long size); } public class ScalesetOperations : StatefulOrm, IScalesetOperations { @@ -111,23 +111,24 @@ public async Async.Task SetSize(Scaleset scaleset, int size) { } } - async Async.Task> SetState(Scaleset scaleSet, ScalesetState state) { - if (scaleSet.State == state) { - return OneFuzzResult.Ok(scaleSet); + public async Async.Task SetState(Scaleset scaleset, ScalesetState state) { + if (scaleset.State == state) { + return scaleset; } - if (scaleSet.State == ScalesetState.Halt) { + if (scaleset.State == ScalesetState.Halt) { // terminal state, unable to change // TODO: should this throw an exception instead? - return OneFuzzResult.Ok(scaleSet); + return scaleset; } - var updatedScaleSet = scaleSet with { State = state }; + var updatedScaleSet = scaleset with { State = state }; var r = await Update(updatedScaleSet); if (!r.IsOk) { var msg = "Failed to update scaleset {scaleSet.ScalesetId} when updating state from {scaleSet.State} to {state}"; _log.Error(msg); - return OneFuzzResult.Error(ErrorCode.UNABLE_TO_UPDATE, msg); + // TODO: this should really return OneFuzzResult but then that propagates up the call stack + throw new Exception(msg); } if (state == ScalesetState.Resize) { @@ -140,18 +141,16 @@ await _context.Events.SendEvent( ); } - return OneFuzzResult.Ok(scaleSet); + return scaleset; } - async Async.Task> SetFailed(Scaleset scaleset, Error error) { + async Async.Task SetFailed(Scaleset scaleset, Error error) { if (scaleset.Error is not null) { - return OneFuzzResult.Ok(scaleset); + // already has an error, don't overwrite it + return scaleset; } var updatedScaleset = await SetState(scaleset with { Error = error }, ScalesetState.CreationFailed); - if (!updatedScaleset.IsOk) { - return updatedScaleset; - } await _context.Events.SendEvent(new EventScalesetFailed(scaleset.ScalesetId, scaleset.PoolName, error)); return updatedScaleset; @@ -190,10 +189,10 @@ public async Async.Task UpdateConfigs(Scaleset scaleSet) { } } - public Async.Task> SetShutdown(Scaleset scaleset, bool now) + public Async.Task SetShutdown(Scaleset scaleset, bool now) => SetState(scaleset, now ? ScalesetState.Halt : ScalesetState.Shutdown); - public async Async.Task> Setup(Scaleset scaleset) { + public async Async.Task Setup(Scaleset scaleset) { //# TODO: How do we pass in SSH configs for Windows? Previously //# This was done as part of the generated per-task setup script. _logTracer.Info($"{SCALESET_LOG_PREFIX} setup. scalset_id: {scaleset.ScalesetId}"); @@ -213,7 +212,7 @@ public async Async.Task> Setup(Scaleset scaleset) { _logTracer.Error($"Failed to save scaleset {scaleset.ScalesetId} due to {r.ErrorV}"); } - return OneFuzzResult.Ok(scaleset); + return scaleset; } if (scaleset.Auth is null) { @@ -275,12 +274,7 @@ public async Async.Task> Setup(Scaleset scaleset) { _logTracer.Error($"Failed to set identity for scaleset {scaleset.ScalesetId} due to: {result.ErrorV}"); return await SetFailed(scaleset, result.ErrorV); } else { - var updateResult = await SetState(scaleset, ScalesetState.Running); - if (!updateResult.IsOk) { - return updateResult.ErrorV; - } - - scaleset = updateResult.OkV; + scaleset = await SetState(scaleset, ScalesetState.Running); } } @@ -289,7 +283,7 @@ public async Async.Task> Setup(Scaleset scaleset) { _logTracer.Error($"Failed to save scale data for scale set: {scaleset.ScalesetId}"); } - return OneFuzzResult.Ok(scaleset); + return scaleset; } @@ -312,7 +306,6 @@ static OneFuzzResult TrySetIdentity(Scaleset scaleset, VirtualMachineS } } - async Async.Task TryEnableAutoScaling(Scaleset scaleset) { _logTracer.Info($"Trying to add auto scaling for scaleset {scaleset.ScalesetId}"); @@ -343,7 +336,7 @@ async Async.Task TryEnableAutoScaling(Scaleset scaleset) { AutoscaleProfile autoScaleProfile; if (autoScaleConfig is null) { - autoScaleProfile = _context.AutoScaleOperations.DeafaultAutoScaleProfile(poolQueueUri!, capacity.Value); + autoScaleProfile = _context.AutoScaleOperations.DefaultAutoScaleProfile(poolQueueUri!, capacity.Value); } else { _logTracer.Info("Using existing auto scale settings from database"); autoScaleProfile = _context.AutoScaleOperations.CreateAutoScaleProfile( @@ -352,9 +345,9 @@ async Async.Task TryEnableAutoScaling(Scaleset scaleset) { autoScaleConfig.Max, autoScaleConfig.Default, autoScaleConfig.ScaleOutAmount, - autoScaleConfig.ScaleOutCoolDown, + autoScaleConfig.ScaleOutCooldown, autoScaleConfig.ScaleInAmount, - autoScaleConfig.ScaleInCoolDown + autoScaleConfig.ScaleInCooldown ); } @@ -364,7 +357,7 @@ async Async.Task TryEnableAutoScaling(Scaleset scaleset) { } - public async Async.Task> Init(Scaleset scaleset) { + public async Async.Task Init(Scaleset scaleset) { _logTracer.Info($"{SCALESET_LOG_PREFIX} init. scaleset_id:{scaleset.ScalesetId}"); var shrinkQueue = new ShrinkQueue(scaleset.ScalesetId, _context.Queue, _logTracer); await shrinkQueue.Create(); @@ -396,7 +389,7 @@ public async Async.Task> Init(Scaleset scaleset) { return await SetState(scaleset, ScalesetState.Setup); } - return OneFuzzResult.Ok(scaleset); + return scaleset; } public async Async.Task Halt(Scaleset scaleset) { @@ -698,10 +691,10 @@ public async Task> GetNodes(Scaleset scaleset) { public IAsyncEnumerable SearchStates(IEnumerable states) => QueryAsync(Query.EqualAnyEnum("state", states)); - public Async.Task> SetSize(Scaleset scaleset, long size) { + public Async.Task SetSize(Scaleset scaleset, long size) { var permittedSize = Math.Min(size, MaxSize(scaleset)); if (permittedSize == scaleset.Size) { - return Async.Task.FromResult(OneFuzzResult.Ok(scaleset)); // nothing to do + return Async.Task.FromResult(scaleset); // nothing to do } scaleset = scaleset with { Size = permittedSize }; diff --git a/src/ApiService/IntegrationTests/Fakes/TestContext.cs b/src/ApiService/IntegrationTests/Fakes/TestContext.cs index d3d6590c3a..9d527fecb6 100644 --- a/src/ApiService/IntegrationTests/Fakes/TestContext.cs +++ b/src/ApiService/IntegrationTests/Fakes/TestContext.cs @@ -113,6 +113,4 @@ public Async.Task InsertAll(params EntityBase[] objs) public ISubnet Subnet => throw new NotImplementedException(); public IImageOperations ImageOperations => throw new NotImplementedException(); - - public IAutoScaleOperations AutoScaleOperations => throw new NotImplementedException(); } From 38edfa21b63a693b2a6a0136b8648bf62523ad96 Mon Sep 17 00:00:00 2001 From: George Pollard Date: Tue, 16 Aug 2022 02:06:15 +0000 Subject: [PATCH 14/14] Format --- src/ApiService/ApiService/OneFuzzTypes/Enums.cs | 2 +- src/ApiService/ApiService/onefuzzlib/VmssOperations.cs | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/ApiService/ApiService/OneFuzzTypes/Enums.cs b/src/ApiService/ApiService/OneFuzzTypes/Enums.cs index a72d817f48..fef13f7376 100644 --- a/src/ApiService/ApiService/OneFuzzTypes/Enums.cs +++ b/src/ApiService/ApiService/OneFuzzTypes/Enums.cs @@ -1,4 +1,4 @@ -using Microsoft.OneFuzz.Service.OneFuzzLib.Orm; +using Microsoft.OneFuzz.Service.OneFuzzLib.Orm; namespace Microsoft.OneFuzz.Service; diff --git a/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs b/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs index ced339aeff..79b808ebcf 100644 --- a/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs +++ b/src/ApiService/ApiService/onefuzzlib/VmssOperations.cs @@ -1,12 +1,12 @@ +using System.Net; using Azure; using Azure.Core; -using System.Net; +using Azure.Data.Tables; using Azure.ResourceManager.Compute; using Azure.ResourceManager.Compute.Models; using Azure.ResourceManager.Models; using Microsoft.Extensions.Caching.Memory; using Microsoft.Rest.Azure; -using Azure.Data.Tables; namespace Microsoft.OneFuzz.Service;